SOLR-5970: Return correct status upon collection creation failure

This commit is contained in:
Ishan Chattopadhyaya 2019-04-28 23:16:29 +05:30
parent 4b49bd99ca
commit dd9899b1c1
7 changed files with 80 additions and 41 deletions

View File

@ -225,6 +225,9 @@ Bug Fixes
* SOLR-12248, SOLR-4647: Grouping is broken on docValues-only fields (non-stored, non-indexed) * SOLR-12248, SOLR-4647: Grouping is broken on docValues-only fields (non-stored, non-indexed)
(Erick Ericsson, Adrien Grand, Munendra S N, Scott Stults, Ishan Chattopadhyaya) (Erick Ericsson, Adrien Grand, Munendra S N, Scott Stults, Ishan Chattopadhyaya)
* SOLR-5970: Return correct status upon collection creation failure (Abraham Elmahrek, Ishan Chattopadhyaya,
Jason Gerlowski, Kesharee Nandan Vishwakarma)
Improvements Improvements
---------------------- ----------------------

View File

@ -301,7 +301,7 @@ public class CreateCollectionCmd implements OverseerCollectionMessageHandler.Cmd
// element, which may be interpreted by the user as a positive ack // element, which may be interpreted by the user as a positive ack
ocmh.cleanupCollection(collectionName, new NamedList<Object>()); ocmh.cleanupCollection(collectionName, new NamedList<Object>());
log.info("Cleaned up artifacts for failed create collection for [{}]", collectionName); log.info("Cleaned up artifacts for failed create collection for [{}]", collectionName);
return; throw new SolrException(ErrorCode.BAD_REQUEST, "Underlying core creation failed while creating collection: " + collectionName);
} else { } else {
log.debug("Finished create command on all shards for collection: {}", collectionName); log.debug("Finished create command on all shards for collection: {}", collectionName);

View File

@ -23,6 +23,7 @@ import static org.hamcrest.CoreMatchers.not;
import java.util.Properties; import java.util.Properties;
import org.apache.solr.client.solrj.impl.CloudSolrClient; import org.apache.solr.client.solrj.impl.CloudSolrClient;
import org.apache.solr.client.solrj.impl.HttpSolrClient;
import org.apache.solr.client.solrj.request.CollectionAdminRequest; import org.apache.solr.client.solrj.request.CollectionAdminRequest;
import org.apache.solr.client.solrj.response.CollectionAdminResponse; import org.apache.solr.client.solrj.response.CollectionAdminResponse;
import org.apache.solr.client.solrj.response.RequestStatusState; import org.apache.solr.client.solrj.response.RequestStatusState;
@ -78,8 +79,9 @@ public class CreateCollectionCleanupTest extends SolrCloudTestCase {
Properties properties = new Properties(); Properties properties = new Properties();
properties.put(CoreAdminParams.DATA_DIR, "/some_invalid_dir/foo"); properties.put(CoreAdminParams.DATA_DIR, "/some_invalid_dir/foo");
create.setProperties(properties); create.setProperties(properties);
CollectionAdminResponse rsp = create.process(cloudClient); expectThrows(HttpSolrClient.RemoteSolrException.class, () -> {
assertFalse(rsp.isSuccess()); CollectionAdminResponse rsp = create.process(cloudClient);
});
// Confirm using LIST that the collection does not exist // Confirm using LIST that the collection does not exist
assertThat("Failed collection is still in the clusterstate: " + cluster.getSolrClient().getClusterStateProvider().getClusterState().getCollectionOrNull(collectionName), assertThat("Failed collection is still in the clusterstate: " + cluster.getSolrClient().getClusterStateProvider().getClusterState().getCollectionOrNull(collectionName),

View File

@ -61,6 +61,7 @@ import org.apache.solr.client.solrj.SolrRequest;
import org.apache.solr.client.solrj.SolrServerException; import org.apache.solr.client.solrj.SolrServerException;
import org.apache.solr.client.solrj.embedded.JettySolrRunner; import org.apache.solr.client.solrj.embedded.JettySolrRunner;
import org.apache.solr.client.solrj.impl.CloudSolrClient; import org.apache.solr.client.solrj.impl.CloudSolrClient;
import org.apache.solr.client.solrj.impl.HttpSolrClient;
import org.apache.solr.client.solrj.impl.HttpClientUtil; import org.apache.solr.client.solrj.impl.HttpClientUtil;
import org.apache.solr.client.solrj.request.ConfigSetAdminRequest; import org.apache.solr.client.solrj.request.ConfigSetAdminRequest;
import org.apache.solr.client.solrj.request.ConfigSetAdminRequest.Create; import org.apache.solr.client.solrj.request.ConfigSetAdminRequest.Create;
@ -89,6 +90,7 @@ import org.apache.zookeeper.KeeperException;
import org.junit.After; import org.junit.After;
import org.junit.Before; import org.junit.Before;
import org.junit.Test; import org.junit.Test;
import static org.junit.matchers.JUnitMatchers.containsString;
import org.noggit.JSONParser; import org.noggit.JSONParser;
import org.noggit.ObjectBuilder; import org.noggit.ObjectBuilder;
import org.slf4j.Logger; import org.slf4j.Logger;
@ -344,29 +346,26 @@ public class TestConfigSetsAPI extends SolrTestCaseJ4 {
@Test @Test
public void testUploadWithScriptUpdateProcessor() throws Exception { public void testUploadWithScriptUpdateProcessor() throws Exception {
for (boolean withAuthorization: Arrays.asList(false, true)) { // Authorization off
String suffix; final String untrustedSuffix = "-untrusted";
if (withAuthorization) { uploadConfigSetWithAssertions("with-script-processor", untrustedSuffix, null, null);
suffix = "-trusted";
protectConfigsHandler();
uploadConfigSetWithAssertions("with-script-processor", suffix, "solr", "SolrRocks");
} else {
suffix = "-untrusted";
uploadConfigSetWithAssertions("with-script-processor", suffix, null, null);
}
// try to create a collection with the uploaded configset // try to create a collection with the uploaded configset
CollectionAdminResponse resp = createCollection("newcollection2", "with-script-processor"+suffix, Throwable thrown = expectThrows(HttpSolrClient.RemoteSolrException.class, () -> {
1, 1, solrCluster.getSolrClient()); createCollection("newcollection2", "with-script-processor" + untrustedSuffix,
1, 1, solrCluster.getSolrClient());
if (withAuthorization) { });
scriptRequest("newcollection2");
} else { assertThat(thrown.getMessage(), containsString("Underlying core creation failed"));
log.info("Client saw errors: "+resp.getErrorMessages());
assertTrue(resp.getErrorMessages() != null && resp.getErrorMessages().size() > 0); // Authorization on
assertTrue(resp.getErrorMessages().getVal(0). final String trustedSuffix = "-trusted";
contains("The configset for this collection was uploaded without any authentication")); protectConfigsHandler();
} uploadConfigSetWithAssertions("with-script-processor", trustedSuffix, "solr", "SolrRocks");
} // try to create a collection with the uploaded configset
CollectionAdminResponse resp = createCollection("newcollection2", "with-script-processor" + trustedSuffix,
1, 1, solrCluster.getSolrClient());
scriptRequest("newcollection2");
} }
protected SolrZkClient zkClient() { protected SolrZkClient zkClient() {

View File

@ -286,19 +286,11 @@ public class CollectionsAPIDistributedZkTest extends SolrCloudTestCase {
String nn1 = cluster.getJettySolrRunner(0).getNodeName(); String nn1 = cluster.getJettySolrRunner(0).getNodeName();
String nn2 = cluster.getJettySolrRunner(1).getNodeName(); String nn2 = cluster.getJettySolrRunner(1).getNodeName();
CollectionAdminResponse resp = CollectionAdminRequest.createCollection("halfcollection", "conf", 2, 1) expectThrows(HttpSolrClient.RemoteSolrException.class, () -> {
.setCreateNodeSet(nn1 + "," + nn2) CollectionAdminResponse resp = CollectionAdminRequest.createCollection("halfcollection", "conf", 2, 1)
.process(cluster.getSolrClient()); .setCreateNodeSet(nn1 + "," + nn2)
.process(cluster.getSolrClient());
SimpleOrderedMap success = (SimpleOrderedMap) resp.getResponse().get("success"); });
SimpleOrderedMap failure = (SimpleOrderedMap) resp.getResponse().get("failure");
assertNotNull(resp.toString(), success);
assertNotNull(resp.toString(), failure);
String val1 = success.getVal(0).toString();
String val2 = failure.getVal(0).toString();
assertTrue(val1.contains("SolrException") || val2.contains("SolrException"));
} }
@Test @Test

View File

@ -25,13 +25,16 @@ import java.util.Map;
import java.util.concurrent.atomic.AtomicLong; import java.util.concurrent.atomic.AtomicLong;
import com.google.common.collect.Lists; import com.google.common.collect.Lists;
import org.apache.solr.cloud.ZkTestServer;
import org.apache.solr.client.solrj.SolrQuery; import org.apache.solr.client.solrj.SolrQuery;
import org.apache.solr.client.solrj.SolrRequest; import org.apache.solr.client.solrj.SolrRequest;
import org.apache.solr.client.solrj.SolrServerException; import org.apache.solr.client.solrj.SolrServerException;
import org.apache.solr.client.solrj.impl.CloudSolrClient; import org.apache.solr.client.solrj.impl.CloudSolrClient;
import org.apache.solr.client.solrj.impl.HttpSolrClient;
import org.apache.solr.client.solrj.impl.HttpSolrClient.RemoteSolrException; import org.apache.solr.client.solrj.impl.HttpSolrClient.RemoteSolrException;
import org.apache.solr.client.solrj.request.CollectionAdminRequest; import org.apache.solr.client.solrj.request.CollectionAdminRequest;
import org.apache.solr.client.solrj.request.QueryRequest; import org.apache.solr.client.solrj.request.QueryRequest;
import org.apache.solr.client.solrj.response.CollectionAdminResponse;
import org.apache.solr.client.solrj.request.V2Request; import org.apache.solr.client.solrj.request.V2Request;
import org.apache.solr.client.solrj.response.QueryResponse; import org.apache.solr.client.solrj.response.QueryResponse;
import org.apache.solr.common.SolrException; import org.apache.solr.common.SolrException;
@ -40,6 +43,7 @@ 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.cloud.Replica;
import org.apache.solr.common.cloud.Slice; import org.apache.solr.common.cloud.Slice;
import org.apache.solr.common.cloud.SolrZkClient;
import org.apache.solr.common.params.CollectionParams; import org.apache.solr.common.params.CollectionParams;
import org.apache.solr.common.params.ModifiableSolrParams; import org.apache.solr.common.params.ModifiableSolrParams;
import org.apache.solr.common.params.ShardParams; import org.apache.solr.common.params.ShardParams;
@ -971,4 +975,34 @@ public class TestCollectionAPI extends ReplicaPropertiesBase {
se.getMessage().toLowerCase(Locale.ROOT).contains("missing required parameter:")); se.getMessage().toLowerCase(Locale.ROOT).contains("missing required parameter:"));
} }
} }
/**
* After a failed attempt to create a collection (due to bad configs), assert that
* the collection can be created with a good collection.
*/
@Test
@ShardsFixed(num = 2)
public void testRecreateCollectionAfterFailure() throws Exception {
// Upload a bad configset
SolrZkClient zkClient = new SolrZkClient(zkServer.getZkHost(), ZkTestServer.TIMEOUT,
ZkTestServer.TIMEOUT, null);
ZkTestServer.putConfig("badconf", zkClient, "/solr", ZkTestServer.SOLRHOME, "bad-error-solrconfig.xml", "solrconfig.xml");
ZkTestServer.putConfig("badconf", zkClient, "/solr", ZkTestServer.SOLRHOME, "schema-minimal.xml", "schema.xml");
zkClient.close();
try (CloudSolrClient client = createCloudClient(null)) {
// first, try creating a collection with badconf
HttpSolrClient.RemoteSolrException rse = expectThrows(HttpSolrClient.RemoteSolrException.class, () -> {
CollectionAdminResponse rsp = CollectionAdminRequest.createCollection
("testcollection", "badconf", 1, 2).process(client);
});
assertNotNull(rse.getMessage());
assertNotSame(0, rse.code());
CollectionAdminResponse rsp = CollectionAdminRequest.createCollection
("testcollection", "conf1", 1, 2).process(client);
assertNull(rsp.getErrorMessages());
assertSame(0, rsp.getStatus());
}
}
} }

View File

@ -788,11 +788,17 @@ public class ZkTestServer {
public static void putConfig(String confName, SolrZkClient zkClient, File solrhome, final String name) public static void putConfig(String confName, SolrZkClient zkClient, File solrhome, final String name)
throws Exception { throws Exception {
putConfig(confName, zkClient, solrhome, name, name); putConfig(confName, zkClient, null, solrhome, name, name);
} }
public static void putConfig(String confName, SolrZkClient zkClient, File solrhome, final String srcName, String destName)
throws Exception { public static void putConfig(String confName, SolrZkClient zkClient, File solrhome, final String srcName,
String destName) throws Exception {
putConfig(confName, zkClient, null, solrhome, srcName, destName);
}
public static void putConfig(String confName, SolrZkClient zkClient, String zkChroot, File solrhome,
final String srcName, String destName) throws Exception {
File file = new File(solrhome, "collection1" File file = new File(solrhome, "collection1"
+ File.separator + "conf" + File.separator + srcName); + File.separator + "conf" + File.separator + srcName);
if (!file.exists()) { if (!file.exists()) {
@ -801,6 +807,9 @@ public class ZkTestServer {
} }
String destPath = "/configs/" + confName + "/" + destName; String destPath = "/configs/" + confName + "/" + destName;
if (zkChroot != null) {
destPath = zkChroot + destPath;
}
log.info("put " + file.getAbsolutePath() + " to " + destPath); log.info("put " + file.getAbsolutePath() + " to " + destPath);
zkClient.makePath(destPath, file, false, true); zkClient.makePath(destPath, file, false, true);
} }