SOLR-12308: LISTALIASES is now assured to return an up-to-date response

* MiniSolrCloudCluster.deleteAllCollections will now first delete aliases
* Minor refactorings to AliasesManager, AliasIntegrationTest, CreateRoutedAliasTest
This commit is contained in:
David Smiley 2018-05-07 22:17:30 -04:00
parent 3e8f31ead0
commit 08ee037ff8
6 changed files with 43 additions and 63 deletions

View File

@ -215,6 +215,10 @@ Bug Fixes
* SOLR-12316: Do not allow to use absolute URIs for including other files in solrconfig.xml and schema parsing. * SOLR-12316: Do not allow to use absolute URIs for including other files in solrconfig.xml and schema parsing.
(Ananthesh, Ishan Chattopadhyaya, Uwe Schindler) (Ananthesh, Ishan Chattopadhyaya, Uwe Schindler)
* SOLR-12308: LISTALIASES is now assured to return an up-to-date response. Also, the test utility
MiniSolrCloudCluster.deleteAllCollections will now first delete aliases since a collection cannot be deleted if an
alias refers to it. (David Smiley)
Optimizations Optimizations
---------------------- ----------------------

View File

@ -590,6 +590,8 @@ public class CollectionsHandler extends RequestHandlerBase implements Permission
*/ */
LISTALIASES_OP(LISTALIASES, (req, rsp, h) -> { LISTALIASES_OP(LISTALIASES, (req, rsp, h) -> {
ZkStateReader zkStateReader = h.coreContainer.getZkController().getZkStateReader(); ZkStateReader zkStateReader = h.coreContainer.getZkController().getZkStateReader();
// if someone calls listAliases, lets ensure we return an up to date response
zkStateReader.aliasesManager.update();
Aliases aliases = zkStateReader.getAliases(); Aliases aliases = zkStateReader.getAliases();
if (aliases != null) { if (aliases != null) {
// the aliases themselves... // the aliases themselves...

View File

@ -83,13 +83,7 @@ public class AliasIntegrationTest extends SolrCloudTestCase {
super.tearDown(); super.tearDown();
IOUtils.close(solrClient, httpClient); IOUtils.close(solrClient, httpClient);
// make sure all aliases created are removed for the next test method cluster.deleteAllCollections(); // note: deletes aliases too
Map<String, String> aliases = new CollectionAdminRequest.ListAliases().process(cluster.getSolrClient()).getAliases();
for (String alias : aliases.keySet()) {
CollectionAdminRequest.deleteAlias(alias).process(cluster.getSolrClient());
}
cluster.deleteAllCollections();
} }
@Test @Test
@ -104,10 +98,9 @@ public class AliasIntegrationTest extends SolrCloudTestCase {
List<String> aliases = zkStateReader.getAliases().resolveAliases("meta1"); List<String> aliases = zkStateReader.getAliases().resolveAliases("meta1");
assertEquals(1, aliases.size()); assertEquals(1, aliases.size());
assertEquals("meta1", aliases.get(0)); assertEquals("meta1", aliases.get(0));
UnaryOperator<Aliases> op6 = a -> a.cloneWithCollectionAlias("meta1", "collection1meta,collection2meta");
final ZkStateReader.AliasesManager aliasesManager = zkStateReader.aliasesManager; final ZkStateReader.AliasesManager aliasesManager = zkStateReader.aliasesManager;
aliasesManager.applyModificationAndExportToZk(op6); aliasesManager.applyModificationAndExportToZk(a -> a.cloneWithCollectionAlias("meta1", "collection1meta,collection2meta"));
aliases = zkStateReader.getAliases().resolveAliases("meta1"); aliases = zkStateReader.getAliases().resolveAliases("meta1");
assertEquals(2, aliases.size()); assertEquals(2, aliases.size());
assertEquals("collection1meta", aliases.get(0)); assertEquals("collection1meta", aliases.get(0));
@ -186,35 +179,28 @@ public class AliasIntegrationTest extends SolrCloudTestCase {
// now check that an independently constructed ZkStateReader can see what we've done. // now check that an independently constructed ZkStateReader can see what we've done.
// i.e. the data is really in zookeeper // i.e. the data is really in zookeeper
String zkAddress = cluster.getZkServer().getZkAddress(); try (SolrZkClient zkClient = new SolrZkClient(cluster.getZkServer().getZkAddress(), 30000)) {
boolean createdZKSR = false;
try(SolrZkClient zkClient = new SolrZkClient(zkAddress, 30000)) {
ZkController.createClusterZkNodes(zkClient); ZkController.createClusterZkNodes(zkClient);
try (ZkStateReader zkStateReader2 = new ZkStateReader(zkClient)) {
zkStateReader2.createClusterStateWatchersAndUpdate();
zkStateReader = new ZkStateReader(zkClient); meta = zkStateReader2.getAliases().getCollectionAliasProperties("meta1");
createdZKSR = true; assertNotNull(meta);
zkStateReader.createClusterStateWatchersAndUpdate();
meta = zkStateReader.getAliases().getCollectionAliasProperties("meta1"); // verify key was removed in independent view
assertNotNull(meta); assertFalse(meta.containsKey("foo"));
// verify key was removed in independent view // but only the specified key was removed
assertFalse(meta.containsKey("foo")); assertTrue(meta.containsKey("foobar"));
assertEquals("bazbam", meta.get("foobar"));
// but only the specified key was removed
assertTrue(meta.containsKey("foobar"));
assertEquals("bazbam", meta.get("foobar"));
Aliases a = zkStateReader.getAliases();
Aliases clone = a.cloneWithCollectionAlias("meta1", null);
meta = clone.getCollectionAliasProperties("meta1");
assertEquals(0,meta.size());
} finally {
if (createdZKSR) {
zkStateReader.close();
} }
} }
// check removal leaves no props behind
assertEquals(0, zkStateReader.getAliases()
.cloneWithCollectionAlias("meta1", null) // not persisted to zk on purpose
.getCollectionAliasProperties("meta1")
.size());
} }
@Test @Test
@ -497,7 +483,7 @@ public class AliasIntegrationTest extends SolrCloudTestCase {
/////////////// ///////////////
CollectionAdminRequest.createAlias("testalias1", "collection1").process(cluster.getSolrClient()); CollectionAdminRequest.createAlias("testalias1", "collection1").process(cluster.getSolrClient());
sleepToAllowZkPropagation();
// ensure that the alias has been registered // ensure that the alias has been registered
assertEquals("collection1", assertEquals("collection1",
new CollectionAdminRequest.ListAliases().process(cluster.getSolrClient()).getAliases().get("testalias1")); new CollectionAdminRequest.ListAliases().process(cluster.getSolrClient()).getAliases().get("testalias1"));

View File

@ -32,7 +32,6 @@ import org.apache.http.entity.ContentType;
import org.apache.http.entity.StringEntity; import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.CloseableHttpClient; import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.util.EntityUtils; import org.apache.http.util.EntityUtils;
import org.apache.lucene.util.IOUtils;
import org.apache.solr.SolrTestCaseJ4; import org.apache.solr.SolrTestCaseJ4;
import org.apache.solr.client.solrj.SolrClient; import org.apache.solr.client.solrj.SolrClient;
import org.apache.solr.client.solrj.SolrServerException; import org.apache.solr.client.solrj.SolrServerException;
@ -57,9 +56,6 @@ import org.junit.Test;
@SolrTestCaseJ4.SuppressSSL @SolrTestCaseJ4.SuppressSSL
public class CreateRoutedAliasTest extends SolrCloudTestCase { public class CreateRoutedAliasTest extends SolrCloudTestCase {
private CloudSolrClient solrClient;
private CloseableHttpClient httpClient;
@BeforeClass @BeforeClass
public static void setupCluster() throws Exception { public static void setupCluster() throws Exception {
configureCluster(2).configure(); configureCluster(2).configure();
@ -73,27 +69,18 @@ public class CreateRoutedAliasTest extends SolrCloudTestCase {
// .process(cluster.getSolrClient()); // .process(cluster.getSolrClient());
} }
@After private CloudSolrClient solrClient;
public void finish() throws Exception {
IOUtils.close(solrClient, httpClient);
}
@Before @Before
public void doBefore() throws Exception { public void doBefore() throws Exception {
solrClient = getCloudSolrClient(cluster); solrClient = getCloudSolrClient(cluster);
httpClient = (CloseableHttpClient) solrClient.getHttpClient(); }
// delete aliases first since they refer to the collections
ZkStateReader zkStateReader = cluster.getSolrClient().getZkStateReader(); @After
//TODO create an API to delete collections attached to the routed alias when the alias is removed public void doAfter() throws Exception {
zkStateReader.aliasesManager.update();// ensure we're seeing the latest cluster.deleteAllCollections(); // deletes aliases too
zkStateReader.aliasesManager.applyModificationAndExportToZk(aliases -> {
Aliases a = zkStateReader.getAliases(); solrClient.close();
for (String alias : a.getCollectionAliasMap().keySet()) {
a = a.cloneWithCollectionAlias(alias,null); // remove
}
return a;
});
cluster.deleteAllCollections();
} }
// This is a fairly complete test where we set many options and see that it both affected the created // This is a fairly complete test where we set many options and see that it both affected the created
@ -235,7 +222,7 @@ public class CreateRoutedAliasTest extends SolrCloudTestCase {
"+30MINUTE", "+30MINUTE",
"evt_dt", "evt_dt",
CollectionAdminRequest.createCollection("_ignored_", "_default", 1, 1) CollectionAdminRequest.createCollection("_ignored_", "_default", 1, 1)
) )
.setTimeZone(TimeZone.getTimeZone("GMT-10")) .setTimeZone(TimeZone.getTimeZone("GMT-10"))
.process(client); .process(client);
} }
@ -366,6 +353,7 @@ public class CreateRoutedAliasTest extends SolrCloudTestCase {
} }
private void assertSuccess(HttpUriRequest msg) throws IOException { private void assertSuccess(HttpUriRequest msg) throws IOException {
CloseableHttpClient httpClient = (CloseableHttpClient) solrClient.getHttpClient();
try (CloseableHttpResponse response = httpClient.execute(msg)) { try (CloseableHttpResponse response = httpClient.execute(msg)) {
if (200 != response.getStatusLine().getStatusCode()) { if (200 != response.getStatusLine().getStatusCode()) {
System.err.println(EntityUtils.toString(response.getEntity())); System.err.println(EntityUtils.toString(response.getEntity()));
@ -375,6 +363,7 @@ public class CreateRoutedAliasTest extends SolrCloudTestCase {
} }
private void assertFailure(HttpUriRequest msg, String expectedErrorSubstring) throws IOException { private void assertFailure(HttpUriRequest msg, String expectedErrorSubstring) throws IOException {
CloseableHttpClient httpClient = (CloseableHttpClient) solrClient.getHttpClient();
try (CloseableHttpResponse response = httpClient.execute(msg)) { try (CloseableHttpResponse response = httpClient.execute(msg)) {
assertEquals(400, response.getStatusLine().getStatusCode()); assertEquals(400, response.getStatusLine().getStatusCode());
String entity = EntityUtils.toString(response.getEntity()); String entity = EntityUtils.toString(response.getEntity());

View File

@ -1685,9 +1685,7 @@ public class ZkStateReader implements Closeable {
public void applyModificationAndExportToZk(UnaryOperator<Aliases> op) { public void applyModificationAndExportToZk(UnaryOperator<Aliases> op) {
final long deadlineNanos = System.nanoTime() + TimeUnit.SECONDS.toNanos(30); final long deadlineNanos = System.nanoTime() + TimeUnit.SECONDS.toNanos(30);
// note: triesLeft tuning is based on ConcurrentCreateRoutedAliasTest // note: triesLeft tuning is based on ConcurrentCreateRoutedAliasTest
int triesLeft = 30; for (int triesLeft = 30; triesLeft > 0; triesLeft--) {
while (triesLeft > 0) {
triesLeft--;
// we could synchronize on "this" but there doesn't seem to be a point; we have a retry loop. // we could synchronize on "this" but there doesn't seem to be a point; we have a retry loop.
Aliases curAliases = getAliases(); Aliases curAliases = getAliases();
Aliases modAliases = op.apply(curAliases); Aliases modAliases = op.apply(curAliases);
@ -1723,9 +1721,7 @@ public class ZkStateReader implements Closeable {
throw new ZooKeeperException(ErrorCode.SERVER_ERROR, e.toString(), e); throw new ZooKeeperException(ErrorCode.SERVER_ERROR, e.toString(), e);
} }
} }
if (triesLeft == 0) { throw new SolrException(ErrorCode.SERVER_ERROR, "Too many successive version failures trying to update aliases");
throw new SolrException(ErrorCode.SERVER_ERROR, "Too many successive version failures trying to update aliases");
}
} }
/** /**
@ -1734,7 +1730,7 @@ public class ZkStateReader implements Closeable {
* @return true if an update was performed * @return true if an update was performed
*/ */
public boolean update() throws KeeperException, InterruptedException { public boolean update() throws KeeperException, InterruptedException {
LOG.debug("Checking ZK for most up to date Aliases " + ALIASES); LOG.debug("Checking ZK for most up to date Aliases {}", ALIASES);
// Call sync() first to ensure the subsequent read (getData) is up to date. // Call sync() first to ensure the subsequent read (getData) is up to date.
zkClient.getSolrZooKeeper().sync(ALIASES, null, null); zkClient.getSolrZooKeeper().sync(ALIASES, null, null);
Stat stat = new Stat(); Stat stat = new Stat();

View File

@ -44,6 +44,7 @@ import org.apache.solr.client.solrj.embedded.SSLConfig;
import org.apache.solr.client.solrj.impl.CloudSolrClient; import org.apache.solr.client.solrj.impl.CloudSolrClient;
import org.apache.solr.client.solrj.impl.CloudSolrClient.Builder; import org.apache.solr.client.solrj.impl.CloudSolrClient.Builder;
import org.apache.solr.client.solrj.request.CollectionAdminRequest; import org.apache.solr.client.solrj.request.CollectionAdminRequest;
import org.apache.solr.common.cloud.Aliases;
import org.apache.solr.common.cloud.Replica; import org.apache.solr.common.cloud.Replica;
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;
@ -444,9 +445,11 @@ public class MiniSolrCloudCluster {
} }
} }
/** Delete all collections (and aliases) */
public void deleteAllCollections() throws Exception { public void deleteAllCollections() throws Exception {
try (ZkStateReader reader = new ZkStateReader(solrClient.getZkStateReader().getZkClient())) { try (ZkStateReader reader = new ZkStateReader(solrClient.getZkStateReader().getZkClient())) {
reader.createClusterStateWatchersAndUpdate(); reader.createClusterStateWatchersAndUpdate(); // up to date aliases & collections
reader.aliasesManager.applyModificationAndExportToZk(aliases -> Aliases.EMPTY);
for (String collection : reader.getClusterState().getCollectionStates().keySet()) { for (String collection : reader.getClusterState().getCollectionStates().keySet()) {
CollectionAdminRequest.deleteCollection(collection).process(solrClient); CollectionAdminRequest.deleteCollection(collection).process(solrClient);
} }