mirror of https://github.com/apache/lucene.git
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:
parent
3e8f31ead0
commit
08ee037ff8
|
@ -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
|
||||||
----------------------
|
----------------------
|
||||||
|
|
||||||
|
|
|
@ -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...
|
||||||
|
|
|
@ -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"));
|
||||||
|
|
|
@ -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());
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue