mirror of https://github.com/apache/lucene.git
SOLR-7577: Add support for rules in CREATESHARD and ADDREPLICA
git-svn-id: https://svn.apache.org/repos/asf/lucene/dev/trunk@1680870 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
961960f1c2
commit
e996532668
|
@ -182,7 +182,8 @@ New Features
|
||||||
* SOLR-7231: DIH-TikaEntityprocessor, create lat-lon field from Metadata
|
* SOLR-7231: DIH-TikaEntityprocessor, create lat-lon field from Metadata
|
||||||
(Tim Allison via Noble Paul)
|
(Tim Allison via Noble Paul)
|
||||||
|
|
||||||
* SOLR-6220: Rule Based Replica Assignment during collection creation (Noble Paul)
|
* SOLR-6220: Rule Based Replica Assignment during collection, shard creation
|
||||||
|
and replica creation (Noble Paul)
|
||||||
|
|
||||||
* SOLR-6968: New 'cardinality' option for stats.field, uses HyperLogLog to efficiently
|
* SOLR-6968: New 'cardinality' option for stats.field, uses HyperLogLog to efficiently
|
||||||
estimate the cardinality of a field w/bounded RAM. (hossman)
|
estimate the cardinality of a field w/bounded RAM. (hossman)
|
||||||
|
|
|
@ -17,6 +17,8 @@ package org.apache.solr.cloud;
|
||||||
* the License.
|
* the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import org.apache.solr.cloud.rule.ReplicaAssigner;
|
||||||
|
import org.apache.solr.cloud.rule.Rule;
|
||||||
import org.apache.solr.common.SolrException;
|
import org.apache.solr.common.SolrException;
|
||||||
import org.apache.solr.common.cloud.ClusterState;
|
import org.apache.solr.common.cloud.ClusterState;
|
||||||
import org.apache.solr.common.cloud.DocCollection;
|
import org.apache.solr.common.cloud.DocCollection;
|
||||||
|
@ -24,6 +26,7 @@ import org.apache.solr.common.cloud.Replica;
|
||||||
import org.apache.solr.common.cloud.Slice;
|
import org.apache.solr.common.cloud.Slice;
|
||||||
import org.apache.solr.common.cloud.ZkStateReader;
|
import org.apache.solr.common.cloud.ZkStateReader;
|
||||||
import org.apache.solr.common.util.StrUtils;
|
import org.apache.solr.common.util.StrUtils;
|
||||||
|
import org.apache.solr.core.CoreContainer;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
@ -32,6 +35,7 @@ import java.util.Collection;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.Comparator;
|
import java.util.Comparator;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
@ -41,6 +45,7 @@ import java.util.regex.Pattern;
|
||||||
import static org.apache.solr.cloud.OverseerCollectionProcessor.CREATE_NODE_SET;
|
import static org.apache.solr.cloud.OverseerCollectionProcessor.CREATE_NODE_SET;
|
||||||
import static org.apache.solr.cloud.OverseerCollectionProcessor.NUM_SLICES;
|
import static org.apache.solr.cloud.OverseerCollectionProcessor.NUM_SLICES;
|
||||||
import static org.apache.solr.common.cloud.ZkStateReader.MAX_SHARDS_PER_NODE;
|
import static org.apache.solr.common.cloud.ZkStateReader.MAX_SHARDS_PER_NODE;
|
||||||
|
import static org.apache.solr.common.cloud.ZkStateReader.REPLICATION_FACTOR;
|
||||||
|
|
||||||
|
|
||||||
public class Assign {
|
public class Assign {
|
||||||
|
@ -129,10 +134,14 @@ public class Assign {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static ArrayList<Node> getNodesForNewShard(ClusterState clusterState, String collectionName, int numSlices, int maxShardsPerNode, int repFactor, String createNodeSetStr) {
|
public static List<Node> getNodesForNewShard(ClusterState clusterState, String collectionName,String shard,int numberOfNodes,
|
||||||
|
String createNodeSetStr, CoreContainer cc) {
|
||||||
|
DocCollection coll = clusterState.getCollection(collectionName);
|
||||||
|
Integer maxShardsPerNode = coll.getInt(MAX_SHARDS_PER_NODE, 1);
|
||||||
|
Integer repFactor = coll.getInt(REPLICATION_FACTOR, 1);
|
||||||
|
int numSlices = coll.getSlices().size();
|
||||||
List<String> createNodeList = createNodeSetStr == null ? null: StrUtils.splitSmart(createNodeSetStr, ",", true);
|
List<String> createNodeList = createNodeSetStr == null ? null: StrUtils.splitSmart(createNodeSetStr, ",", true);
|
||||||
|
|
||||||
|
|
||||||
Set<String> nodes = clusterState.getLiveNodes();
|
Set<String> nodes = clusterState.getLiveNodes();
|
||||||
|
|
||||||
List<String> nodeList = new ArrayList<>(nodes.size());
|
List<String> nodeList = new ArrayList<>(nodes.size());
|
||||||
|
@ -191,14 +200,46 @@ public class Assign {
|
||||||
+ " shards to be created (higher than the allowed number)");
|
+ " shards to be created (higher than the allowed number)");
|
||||||
}
|
}
|
||||||
|
|
||||||
ArrayList<Node> sortedNodeList = new ArrayList<>(nodeNameVsShardCount.values());
|
List l = (List) coll.get(DocCollection.RULE);
|
||||||
Collections.sort(sortedNodeList, new Comparator<Node>() {
|
if(l != null) {
|
||||||
@Override
|
ArrayList<Rule> rules = new ArrayList<>();
|
||||||
public int compare(Node x, Node y) {
|
for (Object o : l) rules.add(new Rule((Map) o));
|
||||||
return (x.weight() < y.weight()) ? -1 : ((x.weight() == y.weight()) ? 0 : 1);
|
Map<String, Map<String,Integer>> shardVsNodes = new LinkedHashMap<>();
|
||||||
|
for (Slice slice : coll.getSlices()) {
|
||||||
|
LinkedHashMap<String, Integer> n = new LinkedHashMap<>();
|
||||||
|
shardVsNodes.put(slice.getName(), n);
|
||||||
|
for (Replica replica : slice.getReplicas()) {
|
||||||
|
Integer count = n.get(replica.getNodeName());
|
||||||
|
if(count == null) count = 0;
|
||||||
|
n.put(replica.getNodeName(),++count);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
List snitches = (List) coll.get(DocCollection.SNITCH);
|
||||||
return sortedNodeList;
|
List<String> nodesList = createNodeList == null ?
|
||||||
|
new ArrayList<>(clusterState.getLiveNodes()) :
|
||||||
|
createNodeList ;
|
||||||
|
Map<ReplicaAssigner.Position, String> positions = new ReplicaAssigner(
|
||||||
|
rules,
|
||||||
|
Collections.singletonMap(shard, numberOfNodes),
|
||||||
|
snitches,
|
||||||
|
shardVsNodes,
|
||||||
|
nodesList, cc, clusterState).getNodeMappings();
|
||||||
|
|
||||||
|
List<Node> n = new ArrayList<>();
|
||||||
|
for (String s : positions.values()) n.add(new Node(s));
|
||||||
|
return n;
|
||||||
|
|
||||||
|
}else {
|
||||||
|
|
||||||
|
ArrayList<Node> sortedNodeList = new ArrayList<>(nodeNameVsShardCount.values());
|
||||||
|
Collections.sort(sortedNodeList, new Comparator<Node>() {
|
||||||
|
@Override
|
||||||
|
public int compare(Node x, Node y) {
|
||||||
|
return (x.weight() < y.weight()) ? -1 : ((x.weight() == y.weight()) ? 0 : 1);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return sortedNodeList;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,6 +18,7 @@ package org.apache.solr.cloud;
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import static org.apache.solr.cloud.Assign.*;
|
import static org.apache.solr.cloud.Assign.*;
|
||||||
|
import static org.apache.solr.common.cloud.DocCollection.SNITCH;
|
||||||
import static org.apache.solr.common.cloud.ZkNodeProps.makeMap;
|
import static org.apache.solr.common.cloud.ZkNodeProps.makeMap;
|
||||||
import static org.apache.solr.common.cloud.ZkStateReader.*;
|
import static org.apache.solr.common.cloud.ZkStateReader.*;
|
||||||
import static org.apache.solr.common.params.CollectionParams.CollectionAction.*;
|
import static org.apache.solr.common.params.CollectionParams.CollectionAction.*;
|
||||||
|
@ -142,7 +143,7 @@ public class OverseerCollectionProcessor implements Runnable, Closeable {
|
||||||
ZkStateReader.MAX_SHARDS_PER_NODE, "1",
|
ZkStateReader.MAX_SHARDS_PER_NODE, "1",
|
||||||
ZkStateReader.AUTO_ADD_REPLICAS, "false",
|
ZkStateReader.AUTO_ADD_REPLICAS, "false",
|
||||||
DocCollection.RULE, null,
|
DocCollection.RULE, null,
|
||||||
DocCollection.SNITCH, null));
|
SNITCH, null));
|
||||||
|
|
||||||
static final Random RANDOM;
|
static final Random RANDOM;
|
||||||
static {
|
static {
|
||||||
|
@ -1296,11 +1297,10 @@ public class OverseerCollectionProcessor implements Runnable, Closeable {
|
||||||
|
|
||||||
ShardHandler shardHandler = shardHandlerFactory.getShardHandler();
|
ShardHandler shardHandler = shardHandlerFactory.getShardHandler();
|
||||||
DocCollection collection = clusterState.getCollection(collectionName);
|
DocCollection collection = clusterState.getCollection(collectionName);
|
||||||
int maxShardsPerNode = collection.getInt(MAX_SHARDS_PER_NODE, 1);
|
|
||||||
int repFactor = message.getInt(REPLICATION_FACTOR, collection.getInt(REPLICATION_FACTOR, 1));
|
int repFactor = message.getInt(REPLICATION_FACTOR, collection.getInt(REPLICATION_FACTOR, 1));
|
||||||
String createNodeSetStr = message.getStr(CREATE_NODE_SET);
|
String createNodeSetStr = message.getStr(CREATE_NODE_SET);
|
||||||
|
List<Node> sortedNodeList = getNodesForNewShard(clusterState, collectionName, sliceName, repFactor,
|
||||||
ArrayList<Node> sortedNodeList = getNodesForNewShard(clusterState, collectionName, numSlices, maxShardsPerNode, repFactor, createNodeSetStr);
|
createNodeSetStr, overseer.getZkController().getCoreContainer());
|
||||||
|
|
||||||
Overseer.getInQueue(zkStateReader.getZkClient()).offer(ZkStateReader.toJSON(message));
|
Overseer.getInQueue(zkStateReader.getZkClient()).offer(ZkStateReader.toJSON(message));
|
||||||
// wait for a while until we see the shard
|
// wait for a while until we see the shard
|
||||||
|
@ -2483,31 +2483,15 @@ public class OverseerCollectionProcessor implements Runnable, Closeable {
|
||||||
Map<String, Integer> sharVsReplicaCount = new HashMap<>();
|
Map<String, Integer> sharVsReplicaCount = new HashMap<>();
|
||||||
|
|
||||||
for (String shard : shardNames) sharVsReplicaCount.put(shard, repFactor);
|
for (String shard : shardNames) sharVsReplicaCount.put(shard, repFactor);
|
||||||
maps = (List<Map>) message.get("snitch");
|
|
||||||
List snitchList = maps == null? Collections.emptyList(): maps;
|
|
||||||
ReplicaAssigner replicaAssigner = new ReplicaAssigner(rules,
|
ReplicaAssigner replicaAssigner = new ReplicaAssigner(rules,
|
||||||
sharVsReplicaCount,
|
sharVsReplicaCount,
|
||||||
snitchList,
|
(List<Map>) message.get(SNITCH),
|
||||||
new HashMap<>(),//this is a new collection. So, there are no nodes in any shard
|
new HashMap<>(),//this is a new collection. So, there are no nodes in any shard
|
||||||
nodeList,
|
nodeList,
|
||||||
overseer.getZkController().getCoreContainer(),
|
overseer.getZkController().getCoreContainer(),
|
||||||
clusterState);
|
clusterState);
|
||||||
|
|
||||||
Map<Position, String> nodeMappings = replicaAssigner.getNodeMappings();
|
return replicaAssigner.getNodeMappings();
|
||||||
if(nodeMappings == null){
|
|
||||||
String msg = "Could not identify nodes matching the rules " + rules ;
|
|
||||||
if(!replicaAssigner.failedNodes.isEmpty()){
|
|
||||||
Map<String, String> failedNodes = new HashMap<>();
|
|
||||||
for (Map.Entry<String, SnitchContext> e : replicaAssigner.failedNodes.entrySet()) {
|
|
||||||
failedNodes.put(e.getKey(), e.getValue().getErrMsg());
|
|
||||||
}
|
|
||||||
msg+=" Some nodes where excluded from assigning replicas because tags could not be obtained from them "+ failedNodes;
|
|
||||||
}
|
|
||||||
msg+= ZkStateReader.toJSONString(replicaAssigner.getNodeVsTags());
|
|
||||||
|
|
||||||
throw new SolrException(ErrorCode.BAD_REQUEST, msg);
|
|
||||||
}
|
|
||||||
return nodeMappings;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private Map<String, Replica> waitToSeeReplicasInState(String collectionName, Collection<String> coreNames) throws InterruptedException {
|
private Map<String, Replica> waitToSeeReplicasInState(String collectionName, Collection<String> coreNames) throws InterruptedException {
|
||||||
|
@ -2560,7 +2544,9 @@ public class OverseerCollectionProcessor implements Runnable, Closeable {
|
||||||
ShardHandler shardHandler = shardHandlerFactory.getShardHandler();
|
ShardHandler shardHandler = shardHandlerFactory.getShardHandler();
|
||||||
|
|
||||||
if (node == null) {
|
if (node == null) {
|
||||||
node = getNodesForNewShard(clusterState, collection, coll.getSlices().size(), coll.getInt(MAX_SHARDS_PER_NODE, 1), coll.getInt(REPLICATION_FACTOR, 1), null).get(0).nodeName;
|
|
||||||
|
node = getNodesForNewShard(clusterState, collection, shard, 1,
|
||||||
|
null, overseer.getZkController().getCoreContainer()).get(0).nodeName;
|
||||||
log.info("Node not provided, Identified {} for creating new replica", node);
|
log.info("Node not provided, Identified {} for creating new replica", node);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -36,6 +36,7 @@ import org.apache.solr.common.cloud.ClusterState;
|
||||||
import org.apache.solr.common.cloud.DocCollection;
|
import org.apache.solr.common.cloud.DocCollection;
|
||||||
import org.apache.solr.common.cloud.Replica;
|
import org.apache.solr.common.cloud.Replica;
|
||||||
import org.apache.solr.common.cloud.Slice;
|
import org.apache.solr.common.cloud.Slice;
|
||||||
|
import org.apache.solr.common.cloud.ZkStateReader;
|
||||||
import org.apache.solr.common.util.StrUtils;
|
import org.apache.solr.common.util.StrUtils;
|
||||||
import org.apache.solr.core.CoreContainer;
|
import org.apache.solr.core.CoreContainer;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
|
@ -55,7 +56,7 @@ public class ReplicaAssigner {
|
||||||
List<Rule> rules;
|
List<Rule> rules;
|
||||||
Map<String, Integer> shardVsReplicaCount;
|
Map<String, Integer> shardVsReplicaCount;
|
||||||
Map<String, Map<String, Object>> nodeVsTags;
|
Map<String, Map<String, Object>> nodeVsTags;
|
||||||
Map<String, Set<String>> shardVsNodes;
|
Map<String, HashMap<String, Integer>> shardVsNodes;
|
||||||
List<String> liveNodes;
|
List<String> liveNodes;
|
||||||
Set<String> tagNames = new HashSet<>();
|
Set<String> tagNames = new HashSet<>();
|
||||||
private Map<String, AtomicInteger> nodeVsCores = new HashMap<>();
|
private Map<String, AtomicInteger> nodeVsCores = new HashMap<>();
|
||||||
|
@ -93,7 +94,7 @@ public class ReplicaAssigner {
|
||||||
public ReplicaAssigner(List<Rule> rules,
|
public ReplicaAssigner(List<Rule> rules,
|
||||||
Map<String, Integer> shardVsReplicaCount,
|
Map<String, Integer> shardVsReplicaCount,
|
||||||
List snitches,
|
List snitches,
|
||||||
Map<String, Set<String>> shardVsNodes,
|
Map<String, Map<String, Integer>> shardVsNodes,
|
||||||
List<String> liveNodes,
|
List<String> liveNodes,
|
||||||
CoreContainer cc, ClusterState clusterState) {
|
CoreContainer cc, ClusterState clusterState) {
|
||||||
this.rules = rules;
|
this.rules = rules;
|
||||||
|
@ -129,6 +130,27 @@ public class ReplicaAssigner {
|
||||||
* the specified rule
|
* the specified rule
|
||||||
*/
|
*/
|
||||||
public Map<Position, String> getNodeMappings() {
|
public Map<Position, String> getNodeMappings() {
|
||||||
|
Map<Position, String> result = getNodeMappings0();
|
||||||
|
if (result == null) {
|
||||||
|
String msg = "Could not identify nodes matching the rules " + rules;
|
||||||
|
if (!failedNodes.isEmpty()) {
|
||||||
|
Map<String, String> failedNodes = new HashMap<>();
|
||||||
|
for (Map.Entry<String, SnitchContext> e : this.failedNodes.entrySet()) {
|
||||||
|
failedNodes.put(e.getKey(), e.getValue().getErrMsg());
|
||||||
|
}
|
||||||
|
msg += " Some nodes where excluded from assigning replicas because tags could not be obtained from them " + failedNodes;
|
||||||
|
}
|
||||||
|
msg += "\n tag values" + ZkStateReader.toJSONString(getNodeVsTags());
|
||||||
|
if (!shardVsNodes.isEmpty()) {
|
||||||
|
msg += "\nInitial state for the coll : " + ZkStateReader.toJSONString(shardVsNodes);
|
||||||
|
}
|
||||||
|
throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, msg);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
Map<Position, String> getNodeMappings0() {
|
||||||
List<String> shardNames = new ArrayList<>(shardVsReplicaCount.keySet());
|
List<String> shardNames = new ArrayList<>(shardVsReplicaCount.keySet());
|
||||||
int[] shardOrder = new int[shardNames.size()];
|
int[] shardOrder = new int[shardNames.size()];
|
||||||
for (int i = 0; i < shardNames.size(); i++) shardOrder[i] = i;
|
for (int i = 0; i < shardNames.size(); i++) shardOrder[i] = i;
|
||||||
|
@ -152,7 +174,6 @@ public class ReplicaAssigner {
|
||||||
result = tryAllPermutations(shardNames, shardOrder, nonWildCardShardRules, true);
|
result = tryAllPermutations(shardNames, shardOrder, nonWildCardShardRules, true);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private Map<Position, String> tryAllPermutations(List<String> shardNames,
|
private Map<Position, String> tryAllPermutations(List<String> shardNames,
|
||||||
|
@ -191,7 +212,7 @@ public class ReplicaAssigner {
|
||||||
Map<String, Map<String, Object>> nodeVsTagsCopy = getDeepCopy(nodeVsTags, 2);
|
Map<String, Map<String, Object>> nodeVsTagsCopy = getDeepCopy(nodeVsTags, 2);
|
||||||
Map<Position, String> result = new LinkedHashMap<>();
|
Map<Position, String> result = new LinkedHashMap<>();
|
||||||
int startPosition = 0;
|
int startPosition = 0;
|
||||||
Map<String, Set<String>> copyOfCurrentState = getDeepCopy(shardVsNodes, 2);
|
Map<String, Map<String, Integer>> copyOfCurrentState = getDeepCopy(shardVsNodes, 2);
|
||||||
List<String> sortedLiveNodes = new ArrayList<>(this.liveNodes);
|
List<String> sortedLiveNodes = new ArrayList<>(this.liveNodes);
|
||||||
Collections.sort(sortedLiveNodes, new Comparator<String>() {
|
Collections.sort(sortedLiveNodes, new Comparator<String>() {
|
||||||
@Override
|
@Override
|
||||||
|
@ -235,9 +256,11 @@ public class ReplicaAssigner {
|
||||||
//We have reached this far means this node can be applied to this position
|
//We have reached this far means this node can be applied to this position
|
||||||
//and all rules are fine. So let us change the currentState
|
//and all rules are fine. So let us change the currentState
|
||||||
result.put(position, liveNode);
|
result.put(position, liveNode);
|
||||||
Set<String> nodeNames = copyOfCurrentState.get(position.shard);
|
Map<String, Integer> nodeNames = copyOfCurrentState.get(position.shard);
|
||||||
if (nodeNames == null) copyOfCurrentState.put(position.shard, nodeNames = new HashSet<>());
|
if (nodeNames == null) copyOfCurrentState.put(position.shard, nodeNames = new HashMap<>());
|
||||||
nodeNames.add(liveNode);
|
Integer n = nodeNames.get(liveNode);
|
||||||
|
n = n == null ? 1 : n + 1;
|
||||||
|
nodeNames.put(liveNode, n);
|
||||||
Number coreCount = (Number) nodeVsTagsCopy.get(liveNode).get(ImplicitSnitch.CORES);
|
Number coreCount = (Number) nodeVsTagsCopy.get(liveNode).get(ImplicitSnitch.CORES);
|
||||||
if (coreCount != null) {
|
if (coreCount != null) {
|
||||||
nodeVsTagsCopy.get(liveNode).put(ImplicitSnitch.CORES, coreCount.intValue() + 1);
|
nodeVsTagsCopy.get(liveNode).put(ImplicitSnitch.CORES, coreCount.intValue() + 1);
|
||||||
|
@ -431,8 +454,8 @@ public class ReplicaAssigner {
|
||||||
|
|
||||||
|
|
||||||
static Map<Class, SnitchInfoImpl> getSnitchInfos(CoreContainer cc, List snitchConf) {
|
static Map<Class, SnitchInfoImpl> getSnitchInfos(CoreContainer cc, List snitchConf) {
|
||||||
|
if (snitchConf == null) snitchConf = Collections.emptyList();
|
||||||
Map<Class, SnitchInfoImpl> snitches = new LinkedHashMap<>();
|
Map<Class, SnitchInfoImpl> snitches = new LinkedHashMap<>();
|
||||||
if (snitchConf == null) return snitches;
|
|
||||||
for (Object o : snitchConf) {
|
for (Object o : snitchConf) {
|
||||||
//instantiating explicitly specified snitches
|
//instantiating explicitly specified snitches
|
||||||
String klas = null;
|
String klas = null;
|
||||||
|
|
|
@ -119,7 +119,7 @@ public class Rule {
|
||||||
* @return MatchStatus
|
* @return MatchStatus
|
||||||
*/
|
*/
|
||||||
MatchStatus tryAssignNodeToShard(String testNode,
|
MatchStatus tryAssignNodeToShard(String testNode,
|
||||||
Map<String, Set<String>> shardVsNodeSet,
|
Map<String, Map<String,Integer>> shardVsNodeSet,
|
||||||
Map<String, Map<String, Object>> nodeVsTags,
|
Map<String, Map<String, Object>> nodeVsTags,
|
||||||
String shardName, Phase phase) {
|
String shardName, Phase phase) {
|
||||||
|
|
||||||
|
@ -153,21 +153,21 @@ public class Rule {
|
||||||
|
|
||||||
private int getNumberOfNodesWithSameTagVal(Condition shardCondition,
|
private int getNumberOfNodesWithSameTagVal(Condition shardCondition,
|
||||||
Map<String, Map<String, Object>> nodeVsTags,
|
Map<String, Map<String, Object>> nodeVsTags,
|
||||||
Map<String, Set<String>> shardVsNodeSet,
|
Map<String, Map<String,Integer>> shardVsNodeSet,
|
||||||
String shardName,
|
String shardName,
|
||||||
Condition tagCondition,
|
Condition tagCondition,
|
||||||
Phase phase) {
|
Phase phase) {
|
||||||
|
|
||||||
int countMatchingThisTagValue = 0;
|
int countMatchingThisTagValue = 0;
|
||||||
for (Map.Entry<String, Set<String>> entry : shardVsNodeSet.entrySet()) {
|
for (Map.Entry<String, Map<String,Integer>> entry : shardVsNodeSet.entrySet()) {
|
||||||
//check if this shard is relevant. either it is a ANY Wild card (**)
|
//check if this shard is relevant. either it is a ANY Wild card (**)
|
||||||
// or this shard is same as the shard in question
|
// or this shard is same as the shard in question
|
||||||
if (shardCondition.val.equals(WILD_WILD_CARD) || entry.getKey().equals(shardName)) {
|
if (shardCondition.val.equals(WILD_WILD_CARD) || entry.getKey().equals(shardName)) {
|
||||||
Set<String> nodesInThisShard = shardVsNodeSet.get(shardCondition.val.equals(WILD_WILD_CARD) ? entry.getKey() : shardName);
|
Map<String,Integer> nodesInThisShard = shardVsNodeSet.get(shardCondition.val.equals(WILD_WILD_CARD) ? entry.getKey() : shardName);
|
||||||
if (nodesInThisShard != null) {
|
if (nodesInThisShard != null) {
|
||||||
for (String aNode : nodesInThisShard) {
|
for (Map.Entry<String,Integer> aNode : nodesInThisShard.entrySet()) {
|
||||||
Object obj = nodeVsTags.get(aNode).get(tag.name);
|
Object obj = nodeVsTags.get(aNode.getKey()).get(tag.name);
|
||||||
if (tagCondition.canMatch(obj, phase)) countMatchingThisTagValue++;
|
if (tagCondition.canMatch(obj, phase)) countMatchingThisTagValue += aNode.getValue();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -177,7 +177,7 @@ public class Rule {
|
||||||
|
|
||||||
public int compare(String n1, String n2,
|
public int compare(String n1, String n2,
|
||||||
Map<String, Map<String, Object>> nodeVsTags,
|
Map<String, Map<String, Object>> nodeVsTags,
|
||||||
Map<String, Set<String>> currentState) {
|
Map<String, Map<String,Integer>> currentState) {
|
||||||
return tag.compare(n1, n2, nodeVsTags);
|
return tag.compare(n1, n2, nodeVsTags);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -29,6 +29,7 @@ import java.util.Set;
|
||||||
|
|
||||||
import org.apache.solr.SolrTestCaseJ4;
|
import org.apache.solr.SolrTestCaseJ4;
|
||||||
import org.apache.solr.cloud.rule.ReplicaAssigner.Position;
|
import org.apache.solr.cloud.rule.ReplicaAssigner.Position;
|
||||||
|
import org.apache.solr.common.SolrException;
|
||||||
import org.apache.solr.common.cloud.ZkStateReader;
|
import org.apache.solr.common.cloud.ZkStateReader;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
|
@ -129,7 +130,7 @@ public class RuleEngineTest extends SolrTestCaseJ4{
|
||||||
mapping = new ReplicaAssigner(
|
mapping = new ReplicaAssigner(
|
||||||
rules,
|
rules,
|
||||||
shardVsReplicaCount, Collections.singletonList(MockSnitch.class.getName()),
|
shardVsReplicaCount, Collections.singletonList(MockSnitch.class.getName()),
|
||||||
new HashMap(), new ArrayList<>(MockSnitch.nodeVsTags.keySet()), null, null).getNodeMappings();
|
new HashMap(), new ArrayList<>(MockSnitch.nodeVsTags.keySet()), null, null).getNodeMappings0();
|
||||||
assertNull(mapping);
|
assertNull(mapping);
|
||||||
|
|
||||||
|
|
||||||
|
@ -141,7 +142,7 @@ public class RuleEngineTest extends SolrTestCaseJ4{
|
||||||
mapping = new ReplicaAssigner(
|
mapping = new ReplicaAssigner(
|
||||||
rules,
|
rules,
|
||||||
shardVsReplicaCount, Collections.singletonList(MockSnitch.class.getName()),
|
shardVsReplicaCount, Collections.singletonList(MockSnitch.class.getName()),
|
||||||
new HashMap(), new ArrayList<>(MockSnitch.nodeVsTags.keySet()), null, null).getNodeMappings();
|
new HashMap(), new ArrayList<>(MockSnitch.nodeVsTags.keySet()), null, null).getNodeMappings0();
|
||||||
assertNotNull(mapping);
|
assertNotNull(mapping);
|
||||||
assertFalse(mapping.containsValue("127.0.0.2:49958_"));
|
assertFalse(mapping.containsValue("127.0.0.2:49958_"));
|
||||||
|
|
||||||
|
@ -164,7 +165,7 @@ public class RuleEngineTest extends SolrTestCaseJ4{
|
||||||
mapping = new ReplicaAssigner(
|
mapping = new ReplicaAssigner(
|
||||||
rules,
|
rules,
|
||||||
shardVsReplicaCount, Collections.singletonList(MockSnitch.class.getName()),
|
shardVsReplicaCount, Collections.singletonList(MockSnitch.class.getName()),
|
||||||
new HashMap(), new ArrayList<>(MockSnitch.nodeVsTags.keySet()), null, null).getNodeMappings();
|
new HashMap(), new ArrayList<>(MockSnitch.nodeVsTags.keySet()), null, null).getNodeMappings0();
|
||||||
assertNull(mapping);
|
assertNull(mapping);
|
||||||
|
|
||||||
rules = parseRules(
|
rules = parseRules(
|
||||||
|
@ -206,7 +207,7 @@ public class RuleEngineTest extends SolrTestCaseJ4{
|
||||||
Map<Position, String> mapping = new ReplicaAssigner(
|
Map<Position, String> mapping = new ReplicaAssigner(
|
||||||
rules,
|
rules,
|
||||||
shardVsReplicaCount, Collections.singletonList(MockSnitch.class.getName()),
|
shardVsReplicaCount, Collections.singletonList(MockSnitch.class.getName()),
|
||||||
new HashMap(), new ArrayList<>(MockSnitch.nodeVsTags.keySet()), null ,null).getNodeMappings();
|
new HashMap(), new ArrayList<>(MockSnitch.nodeVsTags.keySet()), null, null).getNodeMappings0();
|
||||||
assertNull(mapping);
|
assertNull(mapping);
|
||||||
rulesStr = "rack:*,replica:<2~";
|
rulesStr = "rack:*,replica:<2~";
|
||||||
rules = parse(Arrays.asList(rulesStr));
|
rules = parse(Arrays.asList(rulesStr));
|
||||||
|
|
|
@ -29,6 +29,7 @@ import org.apache.solr.client.solrj.response.CollectionAdminResponse;
|
||||||
import org.apache.solr.cloud.AbstractFullDistribZkTestBase;
|
import org.apache.solr.cloud.AbstractFullDistribZkTestBase;
|
||||||
import org.apache.solr.cloud.OverseerCollectionProcessor;
|
import org.apache.solr.cloud.OverseerCollectionProcessor;
|
||||||
import org.apache.solr.common.cloud.DocCollection;
|
import org.apache.solr.common.cloud.DocCollection;
|
||||||
|
import org.apache.solr.common.cloud.ImplicitDocRouter;
|
||||||
import org.apache.solr.common.cloud.ZkStateReader;
|
import org.apache.solr.common.cloud.ZkStateReader;
|
||||||
import org.apache.solr.common.params.ModifiableSolrParams;
|
import org.apache.solr.common.params.ModifiableSolrParams;
|
||||||
import org.apache.solr.core.CoreContainer;
|
import org.apache.solr.core.CoreContainer;
|
||||||
|
@ -43,15 +44,17 @@ public class RulesTest extends AbstractFullDistribZkTestBase {
|
||||||
static final Logger log = LoggerFactory.getLogger(RulesTest.class);
|
static final Logger log = LoggerFactory.getLogger(RulesTest.class);
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
@ShardsFixed(num = 5)
|
||||||
public void doIntegrationTest() throws Exception {
|
public void doIntegrationTest() throws Exception {
|
||||||
String rulesColl = "rulesColl";
|
String rulesColl = "rulesColl";
|
||||||
try (SolrClient client = createNewSolrClient("", getBaseUrl((HttpSolrClient) clients.get(0)))) {
|
try (SolrClient client = createNewSolrClient("", getBaseUrl((HttpSolrClient) clients.get(0)))) {
|
||||||
CollectionAdminResponse rsp;
|
CollectionAdminResponse rsp;
|
||||||
CollectionAdminRequest.Create create = new CollectionAdminRequest.Create();
|
CollectionAdminRequest.Create create = new CollectionAdminRequest.Create();
|
||||||
create.setCollectionName(rulesColl);
|
create.setCollectionName(rulesColl);
|
||||||
create.setNumShards(1);
|
create.setShards("shard1");
|
||||||
|
create.setRouterName(ImplicitDocRouter.NAME);
|
||||||
create.setReplicationFactor(2);
|
create.setReplicationFactor(2);
|
||||||
create.setRule("cores:<4", "node:*,replica:1", "freedisk:>1");
|
create.setRule("cores:<4", "node:*,replica:<2", "freedisk:>1");
|
||||||
create.setSnitch("class:ImplicitSnitch");
|
create.setSnitch("class:ImplicitSnitch");
|
||||||
rsp = create.process(client);
|
rsp = create.process(client);
|
||||||
assertEquals(0, rsp.getStatus());
|
assertEquals(0, rsp.getStatus());
|
||||||
|
@ -63,12 +66,30 @@ public class RulesTest extends AbstractFullDistribZkTestBase {
|
||||||
List list = (List) rulesCollection.get("rule");
|
List list = (List) rulesCollection.get("rule");
|
||||||
assertEquals(3, list.size());
|
assertEquals(3, list.size());
|
||||||
assertEquals ( "<4", ((Map)list.get(0)).get("cores"));
|
assertEquals ( "<4", ((Map)list.get(0)).get("cores"));
|
||||||
assertEquals("1", ((Map) list.get(1)).get("replica"));
|
assertEquals("<2", ((Map) list.get(1)).get("replica"));
|
||||||
assertEquals(">1", ((Map) list.get(2)).get("freedisk"));
|
assertEquals(">1", ((Map) list.get(2)).get("freedisk"));
|
||||||
list = (List) rulesCollection.get("snitch");
|
list = (List) rulesCollection.get("snitch");
|
||||||
assertEquals(1, list.size());
|
assertEquals(1, list.size());
|
||||||
assertEquals ( "ImplicitSnitch", ((Map)list.get(0)).get("class"));
|
assertEquals ( "ImplicitSnitch", ((Map)list.get(0)).get("class"));
|
||||||
|
|
||||||
|
try (SolrClient client = createNewSolrClient("", getBaseUrl((HttpSolrClient) clients.get(0)))) {
|
||||||
|
CollectionAdminResponse rsp;
|
||||||
|
CollectionAdminRequest.CreateShard createShard = new CollectionAdminRequest.CreateShard();
|
||||||
|
createShard.setCollectionName(rulesColl);
|
||||||
|
createShard.setShardName("shard2");
|
||||||
|
rsp = createShard.process(client);
|
||||||
|
assertEquals(0, rsp.getStatus());
|
||||||
|
assertTrue(rsp.isSuccess());
|
||||||
|
|
||||||
|
CollectionAdminRequest.AddReplica addReplica = new CollectionAdminRequest.AddReplica();
|
||||||
|
addReplica.setCollectionName(rulesColl);
|
||||||
|
addReplica.setShardName("shard2");
|
||||||
|
rsp = createShard.process(client);
|
||||||
|
assertEquals(0, rsp.getStatus());
|
||||||
|
assertTrue(rsp.isSuccess());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|
Loading…
Reference in New Issue