SOLR-13674: NodeAddedTrigger does not support configuration of replica type hint.

A new replicaType property has been added to NodeAddTrigger so that new replicas of the given type are added when the preferredOp is addreplica. The default value of replicaType is `NRT`.

This closes #821.

(cherry picked from commit ed137dbe28)
This commit is contained in:
Shalin Shekhar Mangar 2019-08-08 15:14:55 +05:30
parent 8dd116a615
commit de522052c8
7 changed files with 75 additions and 11 deletions

1
.gitignore vendored
View File

@ -28,3 +28,4 @@ pom.xml
.pydevproject .pydevproject
__pycache__ __pycache__
/dev-tools/scripts/scripts.iml /dev-tools/scripts/scripts.iml
.DS_Store

View File

@ -86,6 +86,10 @@ Bug Fixes
* SOLR-13672: Cloud -> Zk Status page now parses response from Zookeeper 3.5.5 correctly (Jörn Franke, janhoy, Shawn Heisey) * SOLR-13672: Cloud -> Zk Status page now parses response from Zookeeper 3.5.5 correctly (Jörn Franke, janhoy, Shawn Heisey)
* SOLR-13674: NodeAddedTrigger does not support configuration of replica type hint. A new replicaType property
has been added to NodeAddTrigger so that new replicas of the given type are added. The default value is `NRT`.
(Irena Shaigorodsky via shalin)
Other Changes Other Changes
---------------------- ----------------------

View File

@ -39,6 +39,7 @@ import org.apache.solr.client.solrj.cloud.autoscaling.UnsupportedSuggester;
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;
import org.apache.solr.common.cloud.Replica;
import org.apache.solr.common.params.AutoScalingParams; import org.apache.solr.common.params.AutoScalingParams;
import org.apache.solr.common.params.CollectionParams; import org.apache.solr.common.params.CollectionParams;
import org.apache.solr.common.params.CoreAdminParams; import org.apache.solr.common.params.CoreAdminParams;
@ -269,6 +270,7 @@ public class ComputePlanAction extends TriggerActionBase {
private Suggester getNodeAddedSuggester(SolrCloudManager cloudManager, Policy.Session session, TriggerEvent event) throws IOException { private Suggester getNodeAddedSuggester(SolrCloudManager cloudManager, Policy.Session session, TriggerEvent event) throws IOException {
String preferredOp = (String) event.getProperty(AutoScalingParams.PREFERRED_OP, CollectionParams.CollectionAction.MOVEREPLICA.toLower()); String preferredOp = (String) event.getProperty(AutoScalingParams.PREFERRED_OP, CollectionParams.CollectionAction.MOVEREPLICA.toLower());
Replica.Type replicaType = (Replica.Type) event.getProperty(AutoScalingParams.REPLICA_TYPE, Replica.Type.NRT);
CollectionParams.CollectionAction action = CollectionParams.CollectionAction.get(preferredOp); CollectionParams.CollectionAction action = CollectionParams.CollectionAction.get(preferredOp);
Suggester suggester = session.getSuggester(action) Suggester suggester = session.getSuggester(action)
@ -289,6 +291,7 @@ public class ComputePlanAction extends TriggerActionBase {
} }
}); });
suggester.hint(Suggester.Hint.COLL_SHARD, collShards); suggester.hint(Suggester.Hint.COLL_SHARD, collShards);
suggester.hint(Suggester.Hint.REPLICATYPE, replicaType);
break; break;
case MOVEREPLICA: case MOVEREPLICA:
case NONE: case NONE:

View File

@ -33,6 +33,7 @@ import java.util.concurrent.TimeUnit;
import org.apache.solr.client.solrj.cloud.SolrCloudManager; import org.apache.solr.client.solrj.cloud.SolrCloudManager;
import org.apache.solr.client.solrj.cloud.autoscaling.TriggerEventType; import org.apache.solr.client.solrj.cloud.autoscaling.TriggerEventType;
import org.apache.solr.common.SolrException; import org.apache.solr.common.SolrException;
import org.apache.solr.common.cloud.Replica;
import org.apache.solr.common.cloud.ZkStateReader; import org.apache.solr.common.cloud.ZkStateReader;
import org.apache.solr.common.params.CollectionParams; import org.apache.solr.common.params.CollectionParams;
import org.apache.solr.core.SolrResourceLoader; import org.apache.solr.core.SolrResourceLoader;
@ -40,6 +41,7 @@ import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import static org.apache.solr.common.params.AutoScalingParams.PREFERRED_OP; import static org.apache.solr.common.params.AutoScalingParams.PREFERRED_OP;
import static org.apache.solr.common.params.AutoScalingParams.REPLICA_TYPE;
/** /**
* Trigger for the {@link TriggerEventType#NODEADDED} event * Trigger for the {@link TriggerEventType#NODEADDED} event
@ -52,10 +54,11 @@ public class NodeAddedTrigger extends TriggerBase {
private Map<String, Long> nodeNameVsTimeAdded = new HashMap<>(); private Map<String, Long> nodeNameVsTimeAdded = new HashMap<>();
private String preferredOp; private String preferredOp;
private Replica.Type replicaType = Replica.Type.NRT;
public NodeAddedTrigger(String name) { public NodeAddedTrigger(String name) {
super(TriggerEventType.NODEADDED, name); super(TriggerEventType.NODEADDED, name);
TriggerUtils.validProperties(validProperties, PREFERRED_OP); TriggerUtils.validProperties(validProperties, PREFERRED_OP, REPLICA_TYPE);
} }
@Override @Override
@ -87,7 +90,14 @@ public class NodeAddedTrigger extends TriggerBase {
super.configure(loader, cloudManager, properties); super.configure(loader, cloudManager, properties);
preferredOp = (String) properties.getOrDefault(PREFERRED_OP, CollectionParams.CollectionAction.MOVEREPLICA.toLower()); preferredOp = (String) properties.getOrDefault(PREFERRED_OP, CollectionParams.CollectionAction.MOVEREPLICA.toLower());
preferredOp = preferredOp.toLowerCase(Locale.ROOT); preferredOp = preferredOp.toLowerCase(Locale.ROOT);
String replicaTypeStr = (String) properties.getOrDefault(REPLICA_TYPE, Replica.Type.NRT.name());
// verify // verify
try {
replicaType = Replica.Type.valueOf(replicaTypeStr);
} catch (IllegalArgumentException | NullPointerException e) {
throw new TriggerValidationException("Unsupported replicaType=" + replicaTypeStr + " specified for node added trigger");
}
CollectionParams.CollectionAction action = CollectionParams.CollectionAction.get(preferredOp); CollectionParams.CollectionAction action = CollectionParams.CollectionAction.get(preferredOp);
switch (action) { switch (action) {
case ADDREPLICA: case ADDREPLICA:
@ -183,7 +193,7 @@ public class NodeAddedTrigger extends TriggerBase {
if (processor != null) { if (processor != null) {
log.debug("NodeAddedTrigger {} firing registered processor for nodes: {} added at times {}, now={}", name, log.debug("NodeAddedTrigger {} firing registered processor for nodes: {} added at times {}, now={}", name,
nodeNames, times, cloudManager.getTimeSource().getTimeNs()); nodeNames, times, cloudManager.getTimeSource().getTimeNs());
if (processor.process(new NodeAddedEvent(getEventType(), getName(), times, nodeNames, preferredOp))) { if (processor.process(new NodeAddedEvent(getEventType(), getName(), times, nodeNames, preferredOp, replicaType))) {
// remove from tracking set only if the fire was accepted // remove from tracking set only if the fire was accepted
nodeNames.forEach(n -> { nodeNames.forEach(n -> {
log.debug("Removing new node from tracking: {}", n); log.debug("Removing new node from tracking: {}", n);
@ -206,12 +216,13 @@ public class NodeAddedTrigger extends TriggerBase {
public static class NodeAddedEvent extends TriggerEvent { public static class NodeAddedEvent extends TriggerEvent {
public NodeAddedEvent(TriggerEventType eventType, String source, List<Long> times, List<String> nodeNames, String preferredOp) { public NodeAddedEvent(TriggerEventType eventType, String source, List<Long> times, List<String> nodeNames, String preferredOp, Replica.Type replicaType) {
// use the oldest time as the time of the event // use the oldest time as the time of the event
super(eventType, source, times.get(0), null); super(eventType, source, times.get(0), null);
properties.put(NODE_NAMES, nodeNames); properties.put(NODE_NAMES, nodeNames);
properties.put(EVENT_TIMES, times); properties.put(EVENT_TIMES, times);
properties.put(PREFERRED_OP, preferredOp); properties.put(PREFERRED_OP, preferredOp);
properties.put(REPLICA_TYPE, replicaType);
} }
} }
} }

View File

@ -524,6 +524,15 @@ public class ComputePlanActionTest extends SolrCloudTestCase {
nodeAddedTriggerWithAddReplicaPreferredOp(collectionNamePrefix, numShards, numCollections); nodeAddedTriggerWithAddReplicaPreferredOp(collectionNamePrefix, numShards, numCollections);
} }
@Test
public void testNodeAddedTriggerWithAddReplicaPreferredOpReplicaType_1Shard() throws Exception {
String collectionNamePrefix = "testNodeAddedTriggerWithAddReplicaPreferredOpReplicaType_1Shard";
int numShards = 1;
int numCollections = 5;
nodeAddedTriggerWithAddReplicaPreferredOpReplicaType(collectionNamePrefix, numShards, numCollections);
}
@Test @Test
// commented out on: 24-Dec-2018 @BadApple(bugUrl="https://issues.apache.org/jira/browse/SOLR-12028") // 14-Oct-2018 // commented out on: 24-Dec-2018 @BadApple(bugUrl="https://issues.apache.org/jira/browse/SOLR-12028") // 14-Oct-2018
public void testNodeAddedTriggerWithAddReplicaPreferredOp_2Shard() throws Exception { public void testNodeAddedTriggerWithAddReplicaPreferredOp_2Shard() throws Exception {
@ -533,9 +542,7 @@ public class ComputePlanActionTest extends SolrCloudTestCase {
nodeAddedTriggerWithAddReplicaPreferredOp(collectionNamePrefix, numShards, numCollections); nodeAddedTriggerWithAddReplicaPreferredOp(collectionNamePrefix, numShards, numCollections);
} }
private void nodeAddedTriggerWithAddReplicaPreferredOp(String collectionNamePrefix, int numShards, int numCollections) throws Exception { private void nodeAddedTriggerWithAddReplicaPreferredOp(String collectionNamePrefix, int numShards, int numCollections) throws Exception {
CloudSolrClient solrClient = cluster.getSolrClient();
String setTriggerCommand = "{" + String setTriggerCommand = "{" +
"'set-trigger' : {" + "'set-trigger' : {" +
"'name' : 'node_added_trigger'," + "'name' : 'node_added_trigger'," +
@ -546,9 +553,6 @@ public class ComputePlanActionTest extends SolrCloudTestCase {
"'actions' : [{'name':'compute_plan', 'class' : 'solr.ComputePlanAction'}," + "'actions' : [{'name':'compute_plan', 'class' : 'solr.ComputePlanAction'}," +
"{'name':'test','class':'" + AssertingTriggerAction.class.getName() + "'}]" + "{'name':'test','class':'" + AssertingTriggerAction.class.getName() + "'}]" +
"}}"; "}}";
SolrRequest req = AutoScalingRequest.create(SolrRequest.METHOD.POST, setTriggerCommand);
NamedList<Object> response = solrClient.request(req);
assertEquals(response.get("result").toString(), "success");
String setClusterPolicyCommand = "{" + String setClusterPolicyCommand = "{" +
" 'set-cluster-policy': [" + " 'set-cluster-policy': [" +
@ -557,13 +561,50 @@ public class ComputePlanActionTest extends SolrCloudTestCase {
" {'nodeRole':'overseer', 'replica':0}" + " {'nodeRole':'overseer', 'replica':0}" +
" ]" + " ]" +
"}"; "}";
nodeAddedTriggerWithAddReplicaPreferredOp(collectionNamePrefix, numShards, numCollections, setTriggerCommand, setClusterPolicyCommand);
}
private void nodeAddedTriggerWithAddReplicaPreferredOpReplicaType(String collectionNamePrefix, int numShards, int numCollections) throws Exception {
String setTriggerCommand = "{" +
"'set-trigger' : {" +
"'name' : 'node_added_trigger'," +
"'event' : 'nodeAdded'," +
"'waitFor' : '1s'," +
"'enabled' : true," +
"'" + AutoScalingParams.PREFERRED_OP + "':'addreplica'," +
"'" + AutoScalingParams.REPLICA_TYPE + "':'" + Replica.Type.PULL + "'," +
"'actions' : [{'name':'compute_plan', 'class' : 'solr.ComputePlanAction'}," +
"{'name':'test','class':'" + AssertingTriggerAction.class.getName() + "'}]" +
"}}";
String setClusterPolicyCommand = "{" +
" 'set-cluster-policy': [" +
" {'cores':'<" + (1 + numCollections * numShards) + "', 'node':'#ANY'}," +
" {'replica':'<2', 'shard': '#EACH', 'node': '#ANY'}," +
" {'nodeRole':'overseer', 'replica':0}" +
" ]" +
"}";
nodeAddedTriggerWithAddReplicaPreferredOp(collectionNamePrefix, numShards, numCollections, setTriggerCommand, setClusterPolicyCommand, 0, 1, 0);
}
private void nodeAddedTriggerWithAddReplicaPreferredOp(String collectionNamePrefix, int numShards, int numCollections, String setTriggerCommand, String setClusterPolicyCommand) throws Exception {
nodeAddedTriggerWithAddReplicaPreferredOp(collectionNamePrefix, numShards, numCollections, setTriggerCommand, setClusterPolicyCommand, 1, null, null);
}
private void nodeAddedTriggerWithAddReplicaPreferredOp(String collectionNamePrefix, int numShards, int numCollections, String setTriggerCommand, String setClusterPolicyCommand, Integer nNrtReplicas, Integer nTlogReplicas, Integer nPullReplicas) throws Exception {
CloudSolrClient solrClient = cluster.getSolrClient();
SolrRequest req = AutoScalingRequest.create(SolrRequest.METHOD.POST, setTriggerCommand);
NamedList<Object> response = solrClient.request(req);
assertEquals(response.get("result").toString(), "success");
req = AutoScalingRequest.create(SolrRequest.METHOD.POST, setClusterPolicyCommand); req = AutoScalingRequest.create(SolrRequest.METHOD.POST, setClusterPolicyCommand);
response = solrClient.request(req); response = solrClient.request(req);
assertEquals(response.get("result").toString(), "success"); assertEquals(response.get("result").toString(), "success");
CollectionAdminRequest.Create create = CollectionAdminRequest.createCollection(collectionNamePrefix + "_0", CollectionAdminRequest.Create create = CollectionAdminRequest.createCollection(collectionNamePrefix + "_0",
"conf", numShards, 1).setMaxShardsPerNode(2); "conf", numShards, nNrtReplicas, nTlogReplicas, nPullReplicas).setMaxShardsPerNode(2);
create.process(solrClient); create.process(solrClient);
waitForState("Timed out waiting for replicas of new collection to be active", waitForState("Timed out waiting for replicas of new collection to be active",

View File

@ -133,6 +133,8 @@ In addition to the parameters described at <<Trigger Configuration>>, this trigg
`preferredOperation`:: (string, optional, defaults to `movereplica`) The operation to be performed in response to an event generated by this trigger. By default, replicas will be moved from other nodes to the added node. The only other supported value is `addreplica` which adds more replicas of the existing collections on the new node. `preferredOperation`:: (string, optional, defaults to `movereplica`) The operation to be performed in response to an event generated by this trigger. By default, replicas will be moved from other nodes to the added node. The only other supported value is `addreplica` which adds more replicas of the existing collections on the new node.
`replicaType`:: (string, optional, defaults to `NRT`) The replica type that will be used to add replicas, in response to an event generated by this trigger, when "preferredOperation" is "ADDREPLICA". By default, the replica(s) will be of type `NRT`. The only other supported values are `PULL` and `TLOG`, which will add more replicas of the specified type to the existing collections on the new node.
.Example: Node Added Trigger to move replicas to new node .Example: Node Added Trigger to move replicas to new node
[source,json] [source,json]
---- ----
@ -145,7 +147,7 @@ In addition to the parameters described at <<Trigger Configuration>>, this trigg
} }
---- ----
.Example: Node Added Trigger to add replicas on new node .Example: Node Added Trigger to add replicas on new node with replica type PULL
[source,json] [source,json]
---- ----
{ {
@ -153,7 +155,8 @@ In addition to the parameters described at <<Trigger Configuration>>, this trigg
"name": "node_added_trigger", "name": "node_added_trigger",
"event": "nodeAdded", "event": "nodeAdded",
"waitFor": "5s", "waitFor": "5s",
"preferredOperation": "ADDREPLICA" "preferredOperation": "ADDREPLICA",
"replicaType": "PULL"
} }
} }
---- ----

View File

@ -50,6 +50,7 @@ public interface AutoScalingParams {
String ABOVE = "above"; String ABOVE = "above";
String BELOW = "below"; String BELOW = "below";
String PREFERRED_OP = "preferredOperation"; String PREFERRED_OP = "preferredOperation";
String REPLICA_TYPE = "replicaType";
// commands // commands
String CMD_SET_TRIGGER = "set-trigger"; String CMD_SET_TRIGGER = "set-trigger";