mirror of https://github.com/apache/lucene.git
Refactor for code clarity, add some comments.
This commit is contained in:
parent
14a988cc2a
commit
ebd409187a
|
@ -332,7 +332,7 @@ public class Assign {
|
||||||
, shard, nrtReplicas, tlogReplicas, pullReplicas, createNodeSet);
|
, shard, nrtReplicas, tlogReplicas, pullReplicas, createNodeSet);
|
||||||
DocCollection coll = clusterState.getCollection(collectionName);
|
DocCollection coll = clusterState.getCollection(collectionName);
|
||||||
int maxShardsPerNode = coll.getMaxShardsPerNode() == -1 ? Integer.MAX_VALUE : coll.getMaxShardsPerNode();
|
int maxShardsPerNode = coll.getMaxShardsPerNode() == -1 ? Integer.MAX_VALUE : coll.getMaxShardsPerNode();
|
||||||
List<String> createNodeList = null;
|
List<String> createNodeList;
|
||||||
|
|
||||||
if (createNodeSet instanceof List) {
|
if (createNodeSet instanceof List) {
|
||||||
createNodeList = (List<String>) createNodeSet;
|
createNodeList = (List<String>) createNodeSet;
|
||||||
|
@ -341,9 +341,13 @@ public class Assign {
|
||||||
createNodeList = createNodeSet == null ? null : new ArrayList<>(new LinkedHashSet<>(StrUtils.splitSmart((String) createNodeSet, ",", true)));
|
createNodeList = createNodeSet == null ? null : new ArrayList<>(new LinkedHashSet<>(StrUtils.splitSmart((String) createNodeSet, ",", true)));
|
||||||
}
|
}
|
||||||
|
|
||||||
HashMap<String, ReplicaCount> nodeNameVsShardCount = getNodeNameVsShardCount(collectionName, clusterState, createNodeList);
|
// produces clear message when down nodes are the root cause, without this the user just
|
||||||
|
// gets a log message of detail about the nodes that are up, and a message that policies could not
|
||||||
|
// be satisfied which then requires study to diagnose the issue.
|
||||||
|
throwIfAnyNotLive(createNodeList,clusterState);
|
||||||
|
|
||||||
if (createNodeList == null) { // We only care if we haven't been told to put new replicas on specific nodes.
|
if (createNodeList == null) { // We only care if we haven't been told to put new replicas on specific nodes.
|
||||||
|
HashMap<String, ReplicaCount> nodeNameVsShardCount = getNodeNameVsShardCount(collectionName, clusterState, null);
|
||||||
long availableSlots = 0;
|
long availableSlots = 0;
|
||||||
for (Map.Entry<String, ReplicaCount> ent : nodeNameVsShardCount.entrySet()) {
|
for (Map.Entry<String, ReplicaCount> ent : nodeNameVsShardCount.entrySet()) {
|
||||||
//ADDREPLICA can put more than maxShardsPerNode on an instance, so this test is necessary.
|
//ADDREPLICA can put more than maxShardsPerNode on an instance, so this test is necessary.
|
||||||
|
@ -411,24 +415,17 @@ public class Assign {
|
||||||
|
|
||||||
static HashMap<String, ReplicaCount> getNodeNameVsShardCount(String collectionName,
|
static HashMap<String, ReplicaCount> getNodeNameVsShardCount(String collectionName,
|
||||||
ClusterState clusterState, List<String> createNodeList) {
|
ClusterState clusterState, List<String> createNodeList) {
|
||||||
Set<String> nodes = clusterState.getLiveNodes();
|
|
||||||
|
|
||||||
List<String> nodeList = new ArrayList<>(nodes.size());
|
|
||||||
nodeList.addAll(nodes);
|
|
||||||
if (createNodeList != null) nodeList.retainAll(createNodeList);
|
|
||||||
|
|
||||||
HashMap<String, ReplicaCount> nodeNameVsShardCount = new HashMap<>();
|
HashMap<String, ReplicaCount> nodeNameVsShardCount = new HashMap<>();
|
||||||
for (String s : nodeList) {
|
for (String s : throwIfAnyNotLive(createNodeList, clusterState)) {
|
||||||
nodeNameVsShardCount.put(s, new ReplicaCount(s));
|
nodeNameVsShardCount.put(s, new ReplicaCount(s));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// if we were given a list, just use that, don't worry about counts
|
||||||
if (createNodeList != null) { // Overrides petty considerations about maxShardsPerNode
|
if (createNodeList != null) { // Overrides petty considerations about maxShardsPerNode
|
||||||
if (createNodeList.size() != nodeNameVsShardCount.size()) {
|
|
||||||
throw new SolrException(SolrException.ErrorCode.BAD_REQUEST,
|
|
||||||
"At least one of the node(s) specified " + createNodeList + " are not currently active in "
|
|
||||||
+ nodeNameVsShardCount.keySet() + ", no action taken.");
|
|
||||||
}
|
|
||||||
return nodeNameVsShardCount;
|
return nodeNameVsShardCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// if we get here we were not given a createNodeList, build a map with real counts.
|
||||||
DocCollection coll = clusterState.getCollection(collectionName);
|
DocCollection coll = clusterState.getCollection(collectionName);
|
||||||
int maxShardsPerNode = coll.getMaxShardsPerNode() == -1 ? Integer.MAX_VALUE : coll.getMaxShardsPerNode();
|
int maxShardsPerNode = coll.getMaxShardsPerNode() == -1 ? Integer.MAX_VALUE : coll.getMaxShardsPerNode();
|
||||||
Map<String, DocCollection> collections = clusterState.getCollectionsMap();
|
Map<String, DocCollection> collections = clusterState.getCollectionsMap();
|
||||||
|
@ -453,6 +450,19 @@ public class Assign {
|
||||||
return nodeNameVsShardCount;
|
return nodeNameVsShardCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static List<String> throwIfAnyNotLive(List<String> createNodeList, ClusterState clusterState) {
|
||||||
|
if (createNodeList != null) {
|
||||||
|
if (!clusterState.getLiveNodes().containsAll(createNodeList)) {
|
||||||
|
throw new SolrException(SolrException.ErrorCode.BAD_REQUEST,
|
||||||
|
"At least one of the node(s) specified " + createNodeList + " are not currently active in "
|
||||||
|
+ createNodeList + ", no action taken.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// the logic that was extracted to this method used to create a defensive copy but no code
|
||||||
|
// was modifying the copy, if this method is made protected or public we want to go back to that
|
||||||
|
return createNodeList; // unmodified, but return for inline use
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Thrown if there is an exception while assigning nodes for replicas
|
* Thrown if there is an exception while assigning nodes for replicas
|
||||||
*/
|
*/
|
||||||
|
@ -550,10 +560,12 @@ public class Assign {
|
||||||
@Override
|
@Override
|
||||||
public List<ReplicaPosition> assign(SolrCloudManager solrCloudManager, AssignRequest assignRequest) throws Assign.AssignmentException, IOException, InterruptedException {
|
public List<ReplicaPosition> assign(SolrCloudManager solrCloudManager, AssignRequest assignRequest) throws Assign.AssignmentException, IOException, InterruptedException {
|
||||||
ClusterState clusterState = solrCloudManager.getClusterStateProvider().getClusterState();
|
ClusterState clusterState = solrCloudManager.getClusterStateProvider().getClusterState();
|
||||||
List<String> nodeList = assignRequest.nodes;
|
List<String> nodeList = assignRequest.nodes; // can this be empty list?
|
||||||
|
|
||||||
HashMap<String, Assign.ReplicaCount> nodeNameVsShardCount = Assign.getNodeNameVsShardCount(assignRequest.collectionName, clusterState, assignRequest.nodes);
|
|
||||||
if (nodeList == null || nodeList.isEmpty()) {
|
if (nodeList == null || nodeList.isEmpty()) {
|
||||||
|
HashMap<String, Assign.ReplicaCount> nodeNameVsShardCount =
|
||||||
|
Assign.getNodeNameVsShardCount(assignRequest.collectionName, clusterState, nodeList);
|
||||||
|
// if nodelist was empty, this map will be empty too. (passing null above however gets a full map)
|
||||||
ArrayList<Assign.ReplicaCount> sortedNodeList = new ArrayList<>(nodeNameVsShardCount.values());
|
ArrayList<Assign.ReplicaCount> sortedNodeList = new ArrayList<>(nodeNameVsShardCount.values());
|
||||||
sortedNodeList.sort(Comparator.comparingInt(Assign.ReplicaCount::weight));
|
sortedNodeList.sort(Comparator.comparingInt(Assign.ReplicaCount::weight));
|
||||||
nodeList = sortedNodeList.stream().map(replicaCount -> replicaCount.nodeName).collect(Collectors.toList());
|
nodeList = sortedNodeList.stream().map(replicaCount -> replicaCount.nodeName).collect(Collectors.toList());
|
||||||
|
@ -561,18 +573,28 @@ public class Assign {
|
||||||
|
|
||||||
int i = 0;
|
int i = 0;
|
||||||
List<ReplicaPosition> result = new ArrayList<>();
|
List<ReplicaPosition> result = new ArrayList<>();
|
||||||
for (String aShard : assignRequest.shardNames)
|
for (String aShard : assignRequest.shardNames) {
|
||||||
for (Map.Entry<Replica.Type, Integer> e : ImmutableMap.of(Replica.Type.NRT, assignRequest.numNrtReplicas,
|
for (Map.Entry<Replica.Type, Integer> e : countsPerReplicaType(assignRequest).entrySet()) {
|
||||||
Replica.Type.TLOG, assignRequest.numTlogReplicas,
|
|
||||||
Replica.Type.PULL, assignRequest.numPullReplicas
|
|
||||||
).entrySet()) {
|
|
||||||
for (int j = 0; j < e.getValue(); j++) {
|
for (int j = 0; j < e.getValue(); j++) {
|
||||||
|
// TODO: not sure if we can receive an non-null, empty assingRequest.nodes,
|
||||||
|
// but if we do this appears to result in a div/0. Also not sure if empty should be
|
||||||
|
// same as null or an error (with better message) here.
|
||||||
result.add(new ReplicaPosition(aShard, j, e.getKey(), nodeList.get(i % nodeList.size())));
|
result.add(new ReplicaPosition(aShard, j, e.getKey(), nodeList.get(i % nodeList.size())));
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// keeps this big ugly construction block out of otherwise legible code
|
||||||
|
private ImmutableMap<Replica.Type, Integer> countsPerReplicaType(AssignRequest assignRequest) {
|
||||||
|
return ImmutableMap.of(
|
||||||
|
Replica.Type.NRT, assignRequest.numNrtReplicas,
|
||||||
|
Replica.Type.TLOG, assignRequest.numTlogReplicas,
|
||||||
|
Replica.Type.PULL, assignRequest.numPullReplicas
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class RulesBasedAssignStrategy implements AssignStrategy {
|
public static class RulesBasedAssignStrategy implements AssignStrategy {
|
||||||
|
|
Loading…
Reference in New Issue