mirror of https://github.com/apache/lucene.git
Harden CollectionPropsTest:
These fixes all relate to testWatcher + testMultipleWatchers: * add additional asserts to the test methods to assert the expected property values are found * mark Watcher.props volatile to prevent stale read by test thread * add some randomization to Watcher.props to either come from the onStateChanged() input or from an explicit call to ZkStateReader.getCollectionProperties - previuosly, for reasons i don't understand, the test only consulted ZkStateReader.getCollectionProperties inside the Watcher, and ignored the onStateChanged() input - now the test validates both * move all Watcher.triggered access into the existing synchronization blocks to prevent waitForTrigger() from returning prematurely due to gaining synch lock _after_ Watcher.triggered was incremented in onStateChanged(), but _before_ onStateChanged() updated Watcher.props * add detailed logging to provide additional info to help debug any additional jenkins failures that might pop up in the future if these fixes aren't sufficient
This commit is contained in:
parent
901f381c61
commit
e8418adedb
|
@ -21,6 +21,8 @@ import java.io.IOException;
|
||||||
import java.lang.invoke.MethodHandles;
|
import java.lang.invoke.MethodHandles;
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.nio.charset.StandardCharsets;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashMap;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
import org.apache.lucene.util.LuceneTestCase;
|
import org.apache.lucene.util.LuceneTestCase;
|
||||||
|
@ -168,27 +170,33 @@ public class CollectionPropsTest extends SolrCloudTestCase {
|
||||||
CollectionProperties collectionProps = new CollectionProperties(zkClient());
|
CollectionProperties collectionProps = new CollectionProperties(zkClient());
|
||||||
|
|
||||||
// Add a watcher to collection props
|
// Add a watcher to collection props
|
||||||
final Watcher watcher = new Watcher();
|
final Watcher watcher = new Watcher("Watcher", random().nextBoolean());
|
||||||
zkStateReader.registerCollectionPropsWatcher(collectionName, watcher);
|
zkStateReader.registerCollectionPropsWatcher(collectionName, watcher);
|
||||||
assertEquals(0, watcher.waitForTrigger(TEST_NIGHTLY?2000:200));
|
assertEquals(0, watcher.waitForTrigger(TEST_NIGHTLY?2000:200));
|
||||||
|
|
||||||
// Trigger a new znode event
|
// Trigger a new znode event
|
||||||
|
log.info("setting value1");
|
||||||
collectionProps.setCollectionProperty(collectionName, "property", "value1");
|
collectionProps.setCollectionProperty(collectionName, "property", "value1");
|
||||||
assertEquals(1, watcher.waitForTrigger());
|
assertEquals(1, watcher.waitForTrigger());
|
||||||
assertEquals("value1", watcher.getProps().get("property"));
|
assertEquals("value1", watcher.getProps().get("property"));
|
||||||
|
|
||||||
// Trigger a value change event
|
// Trigger a value change event
|
||||||
|
log.info("setting value2");
|
||||||
collectionProps.setCollectionProperty(collectionName, "property", "value2");
|
collectionProps.setCollectionProperty(collectionName, "property", "value2");
|
||||||
watcher.waitForTrigger();
|
log.info("(value2) waitForTrigger=={}", watcher.waitForTrigger());
|
||||||
assertEquals("value2", watcher.getProps().get("property"));
|
assertEquals("value2", watcher.getProps().get("property"));
|
||||||
|
|
||||||
// Delete the properties znode
|
// Delete the properties znode
|
||||||
|
log.info("deleting props");
|
||||||
zkStateReader.getZkClient().delete("/collections/" + collectionName + "/collectionprops.json", -1, true);
|
zkStateReader.getZkClient().delete("/collections/" + collectionName + "/collectionprops.json", -1, true);
|
||||||
assertEquals(1, watcher.waitForTrigger());
|
assertEquals(1, watcher.waitForTrigger());
|
||||||
assertTrue(watcher.getProps().isEmpty());
|
final Map<String, String> props = watcher.getProps();
|
||||||
|
assertTrue(props.toString(), props.isEmpty());
|
||||||
|
|
||||||
// Remove watcher and make sure that the watcher is not triggered
|
// Remove watcher and make sure that the watcher is not triggered
|
||||||
|
log.info("removing watcher");
|
||||||
zkStateReader.removeCollectionPropsWatcher(collectionName, watcher);
|
zkStateReader.removeCollectionPropsWatcher(collectionName, watcher);
|
||||||
|
log.info("setting value1 (again)");
|
||||||
collectionProps.setCollectionProperty(collectionName, "property", "value1");
|
collectionProps.setCollectionProperty(collectionName, "property", "value1");
|
||||||
assertEquals("ZK watcher was triggered after it was removed for collection " + collectionName, 0, watcher.waitForTrigger());
|
assertEquals("ZK watcher was triggered after it was removed for collection " + collectionName, 0, watcher.waitForTrigger());
|
||||||
}
|
}
|
||||||
|
@ -202,49 +210,78 @@ public class CollectionPropsTest extends SolrCloudTestCase {
|
||||||
zkStateReader.registerCore(collectionName);
|
zkStateReader.registerCore(collectionName);
|
||||||
|
|
||||||
// Subsequent watchers won't be triggered when adding
|
// Subsequent watchers won't be triggered when adding
|
||||||
final Watcher watcher1 = new Watcher();
|
final Watcher watcher1 = new Watcher("Watcher1", random().nextBoolean());
|
||||||
zkStateReader.registerCollectionPropsWatcher(collectionName, watcher1);
|
zkStateReader.registerCollectionPropsWatcher(collectionName, watcher1);
|
||||||
watcher1.waitForTrigger(); // this might still get triggered because of registerCore
|
watcher1.waitForTrigger(); // this might still get triggered because of registerCore
|
||||||
final Watcher watcher2 = new Watcher();
|
final Watcher watcher2 = new Watcher("Watcher2", random().nextBoolean());
|
||||||
zkStateReader.registerCollectionPropsWatcher(collectionName, watcher2);
|
zkStateReader.registerCollectionPropsWatcher(collectionName, watcher2);
|
||||||
assertEquals(0, watcher2.waitForTrigger(TEST_NIGHTLY?2000:200));
|
assertEquals(0, watcher2.waitForTrigger(TEST_NIGHTLY?2000:200));
|
||||||
|
|
||||||
// Make sure a value change triggers both watchers
|
// Make sure a value change triggers both watchers
|
||||||
|
log.info("setting value1");
|
||||||
collectionProps.setCollectionProperty(collectionName, "property", "value1");
|
collectionProps.setCollectionProperty(collectionName, "property", "value1");
|
||||||
assertEquals(1, watcher1.waitForTrigger());
|
assertEquals(1, watcher1.waitForTrigger());
|
||||||
assertEquals(1, watcher2.waitForTrigger());
|
assertEquals(1, watcher2.waitForTrigger());
|
||||||
|
assertEquals("value1", watcher1.getProps().get("property"));
|
||||||
|
assertEquals("value1", watcher2.getProps().get("property"));
|
||||||
|
|
||||||
// The watchers should be triggered when after the core is unregistered
|
// The watchers should be triggered when after the core is unregistered
|
||||||
zkStateReader.unregisterCore(collectionName);
|
zkStateReader.unregisterCore(collectionName);
|
||||||
|
log.info("setting value2");
|
||||||
collectionProps.setCollectionProperty(collectionName, "property", "value2");
|
collectionProps.setCollectionProperty(collectionName, "property", "value2");
|
||||||
assertEquals(1, watcher1.waitForTrigger());
|
assertEquals(1, watcher1.waitForTrigger());
|
||||||
|
assertEquals(1, watcher2.waitForTrigger());
|
||||||
|
assertEquals("value2", watcher1.getProps().get("property"));
|
||||||
|
assertEquals("value2", watcher2.getProps().get("property"));
|
||||||
|
|
||||||
// The watcher should be triggered after another watcher is removed
|
// The watcher should be triggered after another watcher is removed
|
||||||
|
log.info("removing watcher2");
|
||||||
zkStateReader.removeCollectionPropsWatcher(collectionName, watcher2);
|
zkStateReader.removeCollectionPropsWatcher(collectionName, watcher2);
|
||||||
|
log.info("setting value3");
|
||||||
collectionProps.setCollectionProperty(collectionName, "property", "value3");
|
collectionProps.setCollectionProperty(collectionName, "property", "value3");
|
||||||
assertEquals(1, watcher1.waitForTrigger());
|
assertEquals(1, watcher1.waitForTrigger());
|
||||||
|
assertEquals("value3", watcher1.getProps().get("property"));
|
||||||
|
|
||||||
// The last watcher shouldn't be triggered after removed, even if the core is registered
|
// The last watcher shouldn't be triggered after removed, even if the core is registered
|
||||||
zkStateReader.registerCore(collectionName);
|
zkStateReader.registerCore(collectionName);
|
||||||
|
log.info("removing watcher1");
|
||||||
zkStateReader.removeCollectionPropsWatcher(collectionName, watcher1);
|
zkStateReader.removeCollectionPropsWatcher(collectionName, watcher1);
|
||||||
|
log.info("setting value4");
|
||||||
collectionProps.setCollectionProperty(collectionName, "property", "value4");
|
collectionProps.setCollectionProperty(collectionName, "property", "value4");
|
||||||
assertEquals(0, watcher1.waitForTrigger(TEST_NIGHTLY?2000:200));
|
assertEquals(0, watcher1.waitForTrigger(TEST_NIGHTLY?2000:200));
|
||||||
}
|
}
|
||||||
|
|
||||||
private class Watcher implements CollectionPropsWatcher {
|
private class Watcher implements CollectionPropsWatcher {
|
||||||
private Map<String, String> props = null;
|
private final String name;
|
||||||
private AtomicInteger triggered = new AtomicInteger();
|
private final boolean forceReadPropsFromZk;
|
||||||
|
private volatile Map<String, String> props = Collections.emptyMap();
|
||||||
|
private final AtomicInteger triggered = new AtomicInteger();
|
||||||
|
|
||||||
|
public Watcher(final String name, final boolean forceReadPropsFromZk) {
|
||||||
|
this.name = name;
|
||||||
|
this.forceReadPropsFromZk = forceReadPropsFromZk;
|
||||||
|
log.info("Watcher '{}' initialized with forceReadPropsFromZk={}", name, forceReadPropsFromZk);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean onStateChanged(Map<String, String> collectionProperties) {
|
public boolean onStateChanged(Map<String, String> collectionProperties) {
|
||||||
triggered.incrementAndGet();
|
log.info("{}: state changed...", name);
|
||||||
|
if (forceReadPropsFromZk) {
|
||||||
final ZkStateReader zkStateReader = cluster.getSolrClient().getZkStateReader();
|
final ZkStateReader zkStateReader = cluster.getSolrClient().getZkStateReader();
|
||||||
props = zkStateReader.getCollectionProperties(collectionName);
|
props = Collections.unmodifiableMap(new HashMap(zkStateReader.getCollectionProperties(collectionName)));
|
||||||
|
log.info("{}: Setting props from zk={}", name, props);
|
||||||
|
} else {
|
||||||
|
props = Collections.unmodifiableMap(new HashMap(collectionProperties));
|
||||||
|
log.info("{}: Setting props from caller={}", name, props);
|
||||||
|
}
|
||||||
|
|
||||||
synchronized (this) {
|
synchronized (this) {
|
||||||
|
triggered.incrementAndGet();
|
||||||
|
log.info("{}: notifying", name);
|
||||||
notifyAll();
|
notifyAll();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
log.info("{}: done", name);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -263,9 +300,8 @@ public class CollectionPropsTest extends SolrCloudTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
wait(waitTime);
|
wait(waitTime);
|
||||||
}
|
|
||||||
|
|
||||||
return triggered.getAndSet(0);
|
return triggered.getAndSet(0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue