mirror of https://github.com/apache/lucene.git
SOLR-8677: Restrict creation of shards with invalid names
This commit is contained in:
parent
8bcac81a21
commit
d01230d639
|
@ -266,6 +266,11 @@ Other Changes
|
||||||
* SOLR-5730: Make Lucene's SortingMergePolicy and EarlyTerminatingSortingCollector configurable in Solr.
|
* SOLR-5730: Make Lucene's SortingMergePolicy and EarlyTerminatingSortingCollector configurable in Solr.
|
||||||
(Christine Poerschke, hossmann, Tomás Fernández Löbbe, Shai Erera)
|
(Christine Poerschke, hossmann, Tomás Fernández Löbbe, Shai Erera)
|
||||||
|
|
||||||
|
Other Changes
|
||||||
|
----------------------
|
||||||
|
* SOLR-8677: Prevent shards containing invalid characters from being created. Checks added server-side
|
||||||
|
and in SolrJ. (Shai Erera, Jason Gerlowski, Anshum Gupta)
|
||||||
|
|
||||||
======================= 5.5.0 =======================
|
======================= 5.5.0 =======================
|
||||||
|
|
||||||
Consult the LUCENE_CHANGES.txt file for additional, low level, changes in this release
|
Consult the LUCENE_CHANGES.txt file for additional, low level, changes in this release
|
||||||
|
@ -1018,7 +1023,6 @@ Bug Fixes
|
||||||
|
|
||||||
* SOLR-8355: update permissions were failing node recovery (noble , Anshum Gupta)
|
* SOLR-8355: update permissions were failing node recovery (noble , Anshum Gupta)
|
||||||
|
|
||||||
|
|
||||||
Optimizations
|
Optimizations
|
||||||
----------------------
|
----------------------
|
||||||
|
|
||||||
|
|
|
@ -36,6 +36,7 @@ import com.google.common.collect.ImmutableMap;
|
||||||
import com.google.common.collect.Maps;
|
import com.google.common.collect.Maps;
|
||||||
import org.apache.solr.client.solrj.impl.HttpClientConfigurer;
|
import org.apache.solr.client.solrj.impl.HttpClientConfigurer;
|
||||||
import org.apache.solr.client.solrj.impl.HttpClientUtil;
|
import org.apache.solr.client.solrj.impl.HttpClientUtil;
|
||||||
|
import org.apache.solr.client.solrj.util.SolrIdentifierValidator;
|
||||||
import org.apache.solr.cloud.Overseer;
|
import org.apache.solr.cloud.Overseer;
|
||||||
import org.apache.solr.cloud.ZkController;
|
import org.apache.solr.cloud.ZkController;
|
||||||
import org.apache.solr.common.SolrException;
|
import org.apache.solr.common.SolrException;
|
||||||
|
@ -63,7 +64,6 @@ import org.apache.solr.security.PKIAuthenticationPlugin;
|
||||||
import org.apache.solr.security.SecurityPluginHolder;
|
import org.apache.solr.security.SecurityPluginHolder;
|
||||||
import org.apache.solr.update.UpdateShardHandler;
|
import org.apache.solr.update.UpdateShardHandler;
|
||||||
import org.apache.solr.util.DefaultSolrThreadFactory;
|
import org.apache.solr.util.DefaultSolrThreadFactory;
|
||||||
import org.apache.solr.util.SolrIdentifierValidator;
|
|
||||||
import org.apache.zookeeper.KeeperException;
|
import org.apache.zookeeper.KeeperException;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
@ -804,7 +804,10 @@ public class CoreContainer {
|
||||||
SolrCore core = null;
|
SolrCore core = null;
|
||||||
try {
|
try {
|
||||||
MDCLoggingContext.setCore(core);
|
MDCLoggingContext.setCore(core);
|
||||||
SolrIdentifierValidator.validateCoreName(dcore.getName());
|
if (!SolrIdentifierValidator.validateCoreName(dcore.getName())) {
|
||||||
|
throw new SolrException(ErrorCode.BAD_REQUEST, "Invalid core: " + dcore.getName()
|
||||||
|
+ ". Core names must consist entirely of periods, underscores, and alphanumerics");
|
||||||
|
}
|
||||||
if (zkSys.getZkController() != null) {
|
if (zkSys.getZkController() != null) {
|
||||||
zkSys.getZkController().preRegister(dcore);
|
zkSys.getZkController().preRegister(dcore);
|
||||||
}
|
}
|
||||||
|
@ -1007,7 +1010,10 @@ public class CoreContainer {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void rename(String name, String toName) {
|
public void rename(String name, String toName) {
|
||||||
SolrIdentifierValidator.validateCoreName(toName);
|
if(!SolrIdentifierValidator.validateCoreName(toName)) {
|
||||||
|
throw new SolrException(ErrorCode.BAD_REQUEST, "Invalid core: " + toName
|
||||||
|
+ ". Core names must consist entirely of periods, underscores, and alphanumerics");
|
||||||
|
}
|
||||||
try (SolrCore core = getCore(name)) {
|
try (SolrCore core = getCore(name)) {
|
||||||
if (core != null) {
|
if (core != null) {
|
||||||
registerCore(toName, core, true);
|
registerCore(toName, core, true);
|
||||||
|
|
|
@ -69,6 +69,7 @@ import org.apache.solr.client.solrj.impl.HttpSolrClient;
|
||||||
import org.apache.solr.client.solrj.request.CoreAdminRequest;
|
import org.apache.solr.client.solrj.request.CoreAdminRequest;
|
||||||
import org.apache.solr.client.solrj.request.CoreAdminRequest.RequestSyncShard;
|
import org.apache.solr.client.solrj.request.CoreAdminRequest.RequestSyncShard;
|
||||||
import org.apache.solr.client.solrj.response.RequestStatusState;
|
import org.apache.solr.client.solrj.response.RequestStatusState;
|
||||||
|
import org.apache.solr.client.solrj.util.SolrIdentifierValidator;
|
||||||
import org.apache.solr.cloud.DistributedMap;
|
import org.apache.solr.cloud.DistributedMap;
|
||||||
import org.apache.solr.cloud.Overseer;
|
import org.apache.solr.cloud.Overseer;
|
||||||
import org.apache.solr.cloud.OverseerCollectionMessageHandler;
|
import org.apache.solr.cloud.OverseerCollectionMessageHandler;
|
||||||
|
@ -109,7 +110,6 @@ import org.apache.solr.handler.RequestHandlerBase;
|
||||||
import org.apache.solr.handler.component.ShardHandler;
|
import org.apache.solr.handler.component.ShardHandler;
|
||||||
import org.apache.solr.request.SolrQueryRequest;
|
import org.apache.solr.request.SolrQueryRequest;
|
||||||
import org.apache.solr.response.SolrQueryResponse;
|
import org.apache.solr.response.SolrQueryResponse;
|
||||||
import org.apache.solr.util.SolrIdentifierValidator;
|
|
||||||
import org.apache.zookeeper.CreateMode;
|
import org.apache.zookeeper.CreateMode;
|
||||||
import org.apache.zookeeper.KeeperException;
|
import org.apache.zookeeper.KeeperException;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
|
@ -347,7 +347,16 @@ public class CollectionsHandler extends RequestHandlerBase {
|
||||||
addMapObject(props, SNITCH);
|
addMapObject(props, SNITCH);
|
||||||
verifyRuleParams(h.coreContainer, props);
|
verifyRuleParams(h.coreContainer, props);
|
||||||
final String collectionName = (String) props.get(NAME);
|
final String collectionName = (String) props.get(NAME);
|
||||||
SolrIdentifierValidator.validateCollectionName(collectionName);
|
if (!SolrIdentifierValidator.validateCollectionName(collectionName)) {
|
||||||
|
throw new SolrException(ErrorCode.BAD_REQUEST, "Invalid collection: " + collectionName
|
||||||
|
+ ". Collection names must consist entirely of periods, underscores, and alphanumerics");
|
||||||
|
}
|
||||||
|
final String shardsParam = (String) props.get(SHARDS_PROP);
|
||||||
|
if (StringUtils.isNotEmpty(shardsParam)) {
|
||||||
|
log.info("Validating shards param!!!!!!!!" + shardsParam);
|
||||||
|
verifyShardsParam(shardsParam);
|
||||||
|
log.info("Validating shards param!!!!!!! done" + shardsParam);
|
||||||
|
}
|
||||||
if (SYSTEM_COLL.equals(collectionName)) {
|
if (SYSTEM_COLL.equals(collectionName)) {
|
||||||
//We must always create a .system collection with only a single shard
|
//We must always create a .system collection with only a single shard
|
||||||
props.put(NUM_SLICES, 1);
|
props.put(NUM_SLICES, 1);
|
||||||
|
@ -419,7 +428,10 @@ public class CollectionsHandler extends RequestHandlerBase {
|
||||||
Map<String, Object> call(SolrQueryRequest req, SolrQueryResponse rsp, CollectionsHandler handler)
|
Map<String, Object> call(SolrQueryRequest req, SolrQueryResponse rsp, CollectionsHandler handler)
|
||||||
throws Exception {
|
throws Exception {
|
||||||
final String aliasName = req.getParams().get(NAME);
|
final String aliasName = req.getParams().get(NAME);
|
||||||
SolrIdentifierValidator.validateCollectionName(aliasName);
|
if (!SolrIdentifierValidator.validateCollectionName(aliasName)) {
|
||||||
|
throw new SolrException(ErrorCode.BAD_REQUEST, "Invalid alias: " + aliasName
|
||||||
|
+ ". Aliases must consist entirely of periods, underscores, and alphanumerics");
|
||||||
|
}
|
||||||
return req.getParams().required().getAll(null, NAME, "collections");
|
return req.getParams().required().getAll(null, NAME, "collections");
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -483,6 +495,11 @@ public class CollectionsHandler extends RequestHandlerBase {
|
||||||
COLLECTION_PROP,
|
COLLECTION_PROP,
|
||||||
SHARD_ID_PROP);
|
SHARD_ID_PROP);
|
||||||
ClusterState clusterState = handler.coreContainer.getZkController().getClusterState();
|
ClusterState clusterState = handler.coreContainer.getZkController().getClusterState();
|
||||||
|
final String newShardName = req.getParams().get(SHARD_ID_PROP);
|
||||||
|
if (!SolrIdentifierValidator.validateShardName(newShardName)) {
|
||||||
|
throw new SolrException(ErrorCode.BAD_REQUEST, "Invalid shard: " + newShardName
|
||||||
|
+ ". Shard names must consist entirely of periods, underscores and alphanumerics");
|
||||||
|
}
|
||||||
if (!ImplicitDocRouter.NAME.equals(((Map) clusterState.getCollection(req.getParams().get(COLLECTION_PROP)).get(DOC_ROUTER)).get(NAME)))
|
if (!ImplicitDocRouter.NAME.equals(((Map) clusterState.getCollection(req.getParams().get(COLLECTION_PROP)).get(DOC_ROUTER)).get(NAME)))
|
||||||
throw new SolrException(ErrorCode.BAD_REQUEST, "shards can be added only to 'implicit' collections");
|
throw new SolrException(ErrorCode.BAD_REQUEST, "shards can be added only to 'implicit' collections");
|
||||||
req.getParams().getAll(map,
|
req.getParams().getAll(map,
|
||||||
|
@ -964,6 +981,14 @@ public class CollectionsHandler extends RequestHandlerBase {
|
||||||
return props;
|
return props;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static void verifyShardsParam(String shardsParam) {
|
||||||
|
for (String shard : shardsParam.split(",")) {
|
||||||
|
if (!SolrIdentifierValidator.validateShardName(shard))
|
||||||
|
throw new SolrException(ErrorCode.BAD_REQUEST, "Invalid shard: " + shard
|
||||||
|
+ ". Shard names must consist entirely of periods, underscores and alphanumerics");;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public static final List<String> MODIFIABLE_COLL_PROPS = ImmutableList.of(
|
public static final List<String> MODIFIABLE_COLL_PROPS = ImmutableList.of(
|
||||||
RULE,
|
RULE,
|
||||||
SNITCH,
|
SNITCH,
|
||||||
|
|
|
@ -44,6 +44,7 @@ import org.apache.solr.common.util.NamedList;
|
||||||
import org.apache.zookeeper.KeeperException;
|
import org.apache.zookeeper.KeeperException;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import static org.apache.solr.cloud.OverseerCollectionMessageHandler.ROUTER;
|
||||||
import static org.apache.solr.cloud.OverseerCollectionMessageHandler.SHARD_UNIQUE;
|
import static org.apache.solr.cloud.OverseerCollectionMessageHandler.SHARD_UNIQUE;
|
||||||
|
|
||||||
public class TestCollectionAPI extends ReplicaPropertiesBase {
|
public class TestCollectionAPI extends ReplicaPropertiesBase {
|
||||||
|
@ -79,8 +80,10 @@ public class TestCollectionAPI extends ReplicaPropertiesBase {
|
||||||
replicaPropTest();
|
replicaPropTest();
|
||||||
clusterStatusZNodeVersion();
|
clusterStatusZNodeVersion();
|
||||||
testClusterStateMigration();
|
testClusterStateMigration();
|
||||||
testCollectionCreationNameValidation();
|
testCollectionCreationCollectionNameValidation();
|
||||||
|
testCollectionCreationShardNameValidation();
|
||||||
testAliasCreationNameValidation();
|
testAliasCreationNameValidation();
|
||||||
|
testShardCreationNameValidation();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void clusterStatusWithCollectionAndShard() throws IOException, SolrServerException {
|
private void clusterStatusWithCollectionAndShard() throws IOException, SolrServerException {
|
||||||
|
@ -631,7 +634,7 @@ public class TestCollectionAPI extends ReplicaPropertiesBase {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void testCollectionCreationNameValidation() throws Exception {
|
private void testCollectionCreationCollectionNameValidation() throws Exception {
|
||||||
try (CloudSolrClient client = createCloudClient(null)) {
|
try (CloudSolrClient client = createCloudClient(null)) {
|
||||||
ModifiableSolrParams params = new ModifiableSolrParams();
|
ModifiableSolrParams params = new ModifiableSolrParams();
|
||||||
params.set("action", CollectionParams.CollectionAction.CREATE.toString());
|
params.set("action", CollectionParams.CollectionAction.CREATE.toString());
|
||||||
|
@ -644,9 +647,32 @@ public class TestCollectionAPI extends ReplicaPropertiesBase {
|
||||||
fail();
|
fail();
|
||||||
} catch (RemoteSolrException e) {
|
} catch (RemoteSolrException e) {
|
||||||
final String errorMessage = e.getMessage();
|
final String errorMessage = e.getMessage();
|
||||||
assertTrue(errorMessage.contains("Invalid name"));
|
assertTrue(errorMessage.contains("Invalid collection"));
|
||||||
assertTrue(errorMessage.contains("invalid@name#with$weird%characters"));
|
assertTrue(errorMessage.contains("invalid@name#with$weird%characters"));
|
||||||
assertTrue(errorMessage.contains("Identifiers must consist entirely of"));
|
assertTrue(errorMessage.contains("Collection names must consist entirely of"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void testCollectionCreationShardNameValidation() throws Exception {
|
||||||
|
try (CloudSolrClient client = createCloudClient(null)) {
|
||||||
|
ModifiableSolrParams params = new ModifiableSolrParams();
|
||||||
|
params.set("action", CollectionParams.CollectionAction.CREATE.toString());
|
||||||
|
params.set("name", "valid_collection_name");
|
||||||
|
params.set("router.name", "implicit");
|
||||||
|
params.set("numShards", "1");
|
||||||
|
params.set("shards", "invalid@name#with$weird%characters");
|
||||||
|
SolrRequest request = new QueryRequest(params);
|
||||||
|
request.setPath("/admin/collections");
|
||||||
|
|
||||||
|
try {
|
||||||
|
client.request(request);
|
||||||
|
fail();
|
||||||
|
} catch (RemoteSolrException e) {
|
||||||
|
final String errorMessage = e.getMessage();
|
||||||
|
assertTrue(errorMessage.contains("Invalid shard"));
|
||||||
|
assertTrue(errorMessage.contains("invalid@name#with$weird%characters"));
|
||||||
|
assertTrue(errorMessage.contains("Shard names must consist entirely of"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -665,9 +691,42 @@ public class TestCollectionAPI extends ReplicaPropertiesBase {
|
||||||
fail();
|
fail();
|
||||||
} catch (RemoteSolrException e) {
|
} catch (RemoteSolrException e) {
|
||||||
final String errorMessage = e.getMessage();
|
final String errorMessage = e.getMessage();
|
||||||
assertTrue(errorMessage.contains("Invalid name"));
|
assertTrue(errorMessage.contains("Invalid alias"));
|
||||||
assertTrue(errorMessage.contains("invalid@name#with$weird%characters"));
|
assertTrue(errorMessage.contains("invalid@name#with$weird%characters"));
|
||||||
assertTrue(errorMessage.contains("Identifiers must consist entirely of"));
|
assertTrue(errorMessage.contains("Aliases must consist entirely of"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void testShardCreationNameValidation() throws Exception {
|
||||||
|
try (CloudSolrClient client = createCloudClient(null)) {
|
||||||
|
client.connect();
|
||||||
|
// Create a collection w/ implicit router
|
||||||
|
ModifiableSolrParams params = new ModifiableSolrParams();
|
||||||
|
params.set("action", CollectionParams.CollectionAction.CREATE.toString());
|
||||||
|
params.set("name", "valid_collection_name");
|
||||||
|
params.set("shards", "a");
|
||||||
|
params.set("router.name", "implicit");
|
||||||
|
SolrRequest request = new QueryRequest(params);
|
||||||
|
request.setPath("/admin/collections");
|
||||||
|
client.request(request);
|
||||||
|
|
||||||
|
params = new ModifiableSolrParams();
|
||||||
|
params.set("action", CollectionParams.CollectionAction.CREATESHARD.toString());
|
||||||
|
params.set("collection", "valid_collection_name");
|
||||||
|
params.set("shard", "invalid@name#with$weird%characters");
|
||||||
|
|
||||||
|
request = new QueryRequest(params);
|
||||||
|
request.setPath("/admin/collections");
|
||||||
|
|
||||||
|
try {
|
||||||
|
client.request(request);
|
||||||
|
fail();
|
||||||
|
} catch (RemoteSolrException e) {
|
||||||
|
final String errorMessage = e.getMessage();
|
||||||
|
assertTrue(errorMessage.contains("Invalid shard"));
|
||||||
|
assertTrue(errorMessage.contains("invalid@name#with$weird%characters"));
|
||||||
|
assertTrue(errorMessage.contains("Shard names must consist entirely of"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -139,7 +139,7 @@ public class CoreAdminHandlerTest extends SolrTestCaseJ4 {
|
||||||
resp);
|
resp);
|
||||||
|
|
||||||
} catch (SolrException se) {
|
} catch (SolrException se) {
|
||||||
assertTrue("Expected error message for bad core name.", se.toString().contains("Invalid name"));
|
assertTrue("Expected error message for bad core name.", se.toString().contains("Invalid core"));
|
||||||
}
|
}
|
||||||
CoreDescriptor cd = cores.getCoreDescriptor("ugly$core=name");
|
CoreDescriptor cd = cores.getCoreDescriptor("ugly$core=name");
|
||||||
assertNull("Should NOT have added this core!", cd);
|
assertNull("Should NOT have added this core!", cd);
|
||||||
|
@ -228,7 +228,7 @@ public class CoreAdminHandlerTest extends SolrTestCaseJ4 {
|
||||||
CoreAdminParams.OTHER, "bad$name"),
|
CoreAdminParams.OTHER, "bad$name"),
|
||||||
resp);
|
resp);
|
||||||
} catch (SolrException e) { // why the heck does create return a SolrException (admittedly wrapping an IAE)
|
} catch (SolrException e) { // why the heck does create return a SolrException (admittedly wrapping an IAE)
|
||||||
assertTrue("Expected error message for bad core name.", e.getMessage().contains("Invalid name"));
|
assertTrue("Expected error message for bad core name.", e.getMessage().contains("Invalid core"));
|
||||||
}
|
}
|
||||||
|
|
||||||
cd = cores.getCoreDescriptor("bad$name");
|
cd = cores.getCoreDescriptor("bad$name");
|
||||||
|
|
|
@ -16,9 +16,16 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.client.solrj.request;
|
package org.apache.solr.client.solrj.request;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Properties;
|
||||||
|
|
||||||
import org.apache.solr.client.solrj.SolrClient;
|
import org.apache.solr.client.solrj.SolrClient;
|
||||||
import org.apache.solr.client.solrj.SolrRequest;
|
import org.apache.solr.client.solrj.SolrRequest;
|
||||||
import org.apache.solr.client.solrj.response.CollectionAdminResponse;
|
import org.apache.solr.client.solrj.response.CollectionAdminResponse;
|
||||||
|
import org.apache.solr.client.solrj.util.SolrIdentifierValidator;
|
||||||
import org.apache.solr.common.SolrException;
|
import org.apache.solr.common.SolrException;
|
||||||
import org.apache.solr.common.cloud.DocCollection;
|
import org.apache.solr.common.cloud.DocCollection;
|
||||||
import org.apache.solr.common.cloud.ZkStateReader;
|
import org.apache.solr.common.cloud.ZkStateReader;
|
||||||
|
@ -31,12 +38,6 @@ import org.apache.solr.common.params.ShardParams;
|
||||||
import org.apache.solr.common.params.SolrParams;
|
import org.apache.solr.common.params.SolrParams;
|
||||||
import org.apache.solr.common.util.ContentStream;
|
import org.apache.solr.common.util.ContentStream;
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.util.Collection;
|
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Properties;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class is experimental and subject to change.
|
* This class is experimental and subject to change.
|
||||||
*
|
*
|
||||||
|
@ -122,7 +123,7 @@ public abstract class CollectionAdminRequest <Q extends CollectionAdminRequest<Q
|
||||||
protected abstract static class CollectionSpecificAdminRequest <T extends CollectionAdminRequest<T>> extends CollectionAdminRequest<T> {
|
protected abstract static class CollectionSpecificAdminRequest <T extends CollectionAdminRequest<T>> extends CollectionAdminRequest<T> {
|
||||||
protected String collection = null;
|
protected String collection = null;
|
||||||
|
|
||||||
public final T setCollectionName(String collectionName) {
|
public T setCollectionName(String collectionName) {
|
||||||
this.collection = collectionName;
|
this.collection = collectionName;
|
||||||
return getThis();
|
return getThis();
|
||||||
}
|
}
|
||||||
|
@ -277,7 +278,6 @@ public abstract class CollectionAdminRequest <Q extends CollectionAdminRequest<Q
|
||||||
public Create setConfigName(String config) { this.configName = config; return this; }
|
public Create setConfigName(String config) { this.configName = config; return this; }
|
||||||
public Create setCreateNodeSet(String nodeSet) { this.createNodeSet = nodeSet; return this; }
|
public Create setCreateNodeSet(String nodeSet) { this.createNodeSet = nodeSet; return this; }
|
||||||
public Create setRouterName(String routerName) { this.routerName = routerName; return this; }
|
public Create setRouterName(String routerName) { this.routerName = routerName; return this; }
|
||||||
public Create setShards(String shards) { this.shards = shards; return this; }
|
|
||||||
public Create setRouterField(String routerField) { this.routerField = routerField; return this; }
|
public Create setRouterField(String routerField) { this.routerField = routerField; return this; }
|
||||||
public Create setNumShards(Integer numShards) {this.numShards = numShards; return this; }
|
public Create setNumShards(Integer numShards) {this.numShards = numShards; return this; }
|
||||||
public Create setMaxShardsPerNode(Integer numShards) { this.maxShardsPerNode = numShards; return this; }
|
public Create setMaxShardsPerNode(Integer numShards) { this.maxShardsPerNode = numShards; return this; }
|
||||||
|
@ -297,6 +297,41 @@ public abstract class CollectionAdminRequest <Q extends CollectionAdminRequest<Q
|
||||||
public Boolean getAutoAddReplicas() { return autoAddReplicas; }
|
public Boolean getAutoAddReplicas() { return autoAddReplicas; }
|
||||||
public Integer getStateFormat() { return stateFormat; }
|
public Integer getStateFormat() { return stateFormat; }
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provide the name of the shards to be created, separated by commas
|
||||||
|
*
|
||||||
|
* Shard names must consist entirely of periods, underscores and alphanumerics. Other characters are not allowed.
|
||||||
|
*
|
||||||
|
* @throws IllegalArgumentException if any of the shard names contain invalid characters.
|
||||||
|
*/
|
||||||
|
public Create setShards(String shards) {
|
||||||
|
for (String shard : shards.split(",")) {
|
||||||
|
if (!SolrIdentifierValidator.validateShardName(shard)) {
|
||||||
|
throw new IllegalArgumentException("Invalid shard: " + shard
|
||||||
|
+ ". Shard names must consist entirely of periods, underscores and alphanumerics");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.shards = shards;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provide the name of the collection to be created.
|
||||||
|
*
|
||||||
|
* Collection names must consist entirely of periods, underscores and alphanumerics. Other characters are not allowed.
|
||||||
|
*
|
||||||
|
* @throws IllegalArgumentException if the collection name contains invalid characters.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public Create setCollectionName(String collectionName) throws SolrException {
|
||||||
|
if (!SolrIdentifierValidator.validateCollectionName(collectionName)) {
|
||||||
|
throw new IllegalArgumentException("Invalid collection: " + collectionName
|
||||||
|
+ ". Collection names must consist entirely of periods, underscores, and alphanumerics");
|
||||||
|
}
|
||||||
|
this.collection = collectionName;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
public Properties getProperties() {
|
public Properties getProperties() {
|
||||||
return properties;
|
return properties;
|
||||||
}
|
}
|
||||||
|
@ -410,6 +445,23 @@ public abstract class CollectionAdminRequest <Q extends CollectionAdminRequest<Q
|
||||||
action = CollectionAction.CREATESHARD;
|
action = CollectionAction.CREATESHARD;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provide the name of the shard to be created.
|
||||||
|
*
|
||||||
|
* Shard names must consist entirely of periods, underscores and alphanumerics. Other characters are not allowed.
|
||||||
|
*
|
||||||
|
* @throws IllegalArgumentException if the shard name contains invalid characters.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public CreateShard setShardName(String shardName) {
|
||||||
|
if (!SolrIdentifierValidator.validateShardName(shardName)) {
|
||||||
|
throw new IllegalArgumentException("Invalid shard: " + shardName
|
||||||
|
+ ". Shard names must consist entirely of periods, underscores and alphanumerics");
|
||||||
|
}
|
||||||
|
this.shardName = shardName;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public SolrParams getParams() {
|
public SolrParams getParams() {
|
||||||
ModifiableSolrParams params = (ModifiableSolrParams) super.getParams();
|
ModifiableSolrParams params = (ModifiableSolrParams) super.getParams();
|
||||||
|
@ -588,7 +640,18 @@ public abstract class CollectionAdminRequest <Q extends CollectionAdminRequest<Q
|
||||||
action = CollectionAction.CREATEALIAS;
|
action = CollectionAction.CREATEALIAS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provide the name of the alias to be created.
|
||||||
|
*
|
||||||
|
* Alias names must consist entirely of periods, underscores and alphanumerics. Other characters are not allowed.
|
||||||
|
*
|
||||||
|
* @throws IllegalArgumentException if the alias name contains invalid characters.
|
||||||
|
*/
|
||||||
public CreateAlias setAliasName(String aliasName) {
|
public CreateAlias setAliasName(String aliasName) {
|
||||||
|
if (!SolrIdentifierValidator.validateCollectionName(aliasName)) {
|
||||||
|
throw new IllegalArgumentException("Invalid alias: " + aliasName
|
||||||
|
+ ". Aliases must consist entirely of periods, underscores, and alphanumerics");
|
||||||
|
}
|
||||||
this.aliasName = aliasName;
|
this.aliasName = aliasName;
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,6 +20,7 @@ import org.apache.solr.client.solrj.SolrClient;
|
||||||
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.response.CoreAdminResponse;
|
import org.apache.solr.client.solrj.response.CoreAdminResponse;
|
||||||
|
import org.apache.solr.client.solrj.util.SolrIdentifierValidator;
|
||||||
import org.apache.solr.common.cloud.Replica;
|
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.CoreAdminParams;
|
import org.apache.solr.common.params.CoreAdminParams;
|
||||||
|
@ -100,6 +101,22 @@ public class CoreAdminRequest extends SolrRequest<CoreAdminResponse> {
|
||||||
public Boolean getIsTransient() { return isTransient; }
|
public Boolean getIsTransient() { return isTransient; }
|
||||||
public String getCollectionConfigName() { return collectionConfigName;}
|
public String getCollectionConfigName() { return collectionConfigName;}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provide the name of the core to be created.
|
||||||
|
*
|
||||||
|
* Core names must consist entirely of periods, underscores and alphanumerics. Other characters are not allowed.
|
||||||
|
*
|
||||||
|
* @throws IllegalArgumentException if the core name contains invalid characters.
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
public void setCoreName(String coreName) {
|
||||||
|
if (!SolrIdentifierValidator.validateCoreName(coreName)) {
|
||||||
|
throw new IllegalArgumentException("Invalid collection: " + coreName
|
||||||
|
+ ". Core names must consist entirely of periods, underscores, and alphanumerics");
|
||||||
|
}
|
||||||
|
this.core = coreName;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public SolrParams getParams() {
|
public SolrParams getParams() {
|
||||||
if( action == null ) {
|
if( action == null ) {
|
||||||
|
@ -450,7 +467,7 @@ public class CoreAdminRequest extends SolrRequest<CoreAdminResponse> {
|
||||||
super( METHOD.GET, path );
|
super( METHOD.GET, path );
|
||||||
}
|
}
|
||||||
|
|
||||||
public final void setCoreName( String coreName )
|
public void setCoreName( String coreName )
|
||||||
{
|
{
|
||||||
this.core = coreName;
|
this.core = coreName;
|
||||||
}
|
}
|
||||||
|
@ -535,8 +552,18 @@ public class CoreAdminRequest extends SolrRequest<CoreAdminResponse> {
|
||||||
return req.process(client);
|
return req.process(client);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static CoreAdminResponse renameCore(String coreName, String newName, SolrClient client ) throws SolrServerException, IOException
|
/**
|
||||||
{
|
* Rename an existing core.
|
||||||
|
*
|
||||||
|
* @throws IllegalArgumentException if the new core name contains invalid characters.
|
||||||
|
*/
|
||||||
|
public static CoreAdminResponse renameCore(String coreName, String newName, SolrClient client )
|
||||||
|
throws SolrServerException, IOException {
|
||||||
|
if (!SolrIdentifierValidator.validateCoreName(newName)) {
|
||||||
|
throw new IllegalArgumentException("Invalid collection: " + newName
|
||||||
|
+ ". Core names must consist entirely of periods, underscores, and alphanumerics");
|
||||||
|
}
|
||||||
|
|
||||||
CoreAdminRequest req = new CoreAdminRequest();
|
CoreAdminRequest req = new CoreAdminRequest();
|
||||||
req.setCoreName(coreName);
|
req.setCoreName(coreName);
|
||||||
req.setOtherCoreName(newName);
|
req.setOtherCoreName(newName);
|
||||||
|
|
|
@ -1,3 +1,7 @@
|
||||||
|
package org.apache.solr.client.solrj.util;
|
||||||
|
|
||||||
|
import java.util.regex.Pattern;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Licensed to the Apache Software Foundation (ASF) under one or more
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
* contributor license agreements. See the NOTICE file distributed with
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
@ -14,14 +18,6 @@
|
||||||
* See the License for the specific language governing permissions and
|
* See the License for the specific language governing permissions and
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.util;
|
|
||||||
|
|
||||||
import java.lang.invoke.MethodHandles;
|
|
||||||
import java.util.regex.Pattern;
|
|
||||||
|
|
||||||
import org.apache.solr.common.SolrException;
|
|
||||||
import org.slf4j.Logger;
|
|
||||||
import org.slf4j.LoggerFactory;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Ensures that provided identifiers align with Solr's recommendations/requirements for choosing
|
* Ensures that provided identifiers align with Solr's recommendations/requirements for choosing
|
||||||
|
@ -30,18 +26,26 @@ import org.slf4j.LoggerFactory;
|
||||||
* Identifiers are allowed to contain underscores, periods, and alphanumeric characters.
|
* Identifiers are allowed to contain underscores, periods, and alphanumeric characters.
|
||||||
*/
|
*/
|
||||||
public class SolrIdentifierValidator {
|
public class SolrIdentifierValidator {
|
||||||
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
|
|
||||||
final static Pattern identifierPattern = Pattern.compile("^[\\._A-Za-z0-9]*$");
|
final static Pattern identifierPattern = Pattern.compile("^[\\._A-Za-z0-9]*$");
|
||||||
|
|
||||||
public static void validateCollectionName(String collectionName) throws SolrException {
|
public static boolean validateShardName(String shardName) {
|
||||||
validateCoreName(collectionName);
|
return validateIdentifier(shardName);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void validateCoreName(String name) throws SolrException {
|
public static boolean validateCollectionName(String collectionName) {
|
||||||
if (name == null || !identifierPattern.matcher(name).matches()) {
|
return validateIdentifier(collectionName);
|
||||||
log.info("Validation failed on the invalid identifier [{}]. Throwing SolrException to indicate a BAD REQUEST.", name);
|
}
|
||||||
throw new SolrException(SolrException.ErrorCode.BAD_REQUEST,
|
|
||||||
"Invalid name: '" + name + "' Identifiers must consist entirely of periods, underscores and alphanumerics");
|
public static boolean validateCoreName(String name) {
|
||||||
}
|
return validateIdentifier(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean validateIdentifier(String identifier) {
|
||||||
|
if (identifier == null || ! identifierPattern.matcher(identifier).matches()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,85 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
package org.apache.solr.client.solrj.request;
|
||||||
|
|
||||||
|
import org.apache.lucene.util.LuceneTestCase;
|
||||||
|
import org.apache.solr.client.solrj.request.CollectionAdminRequest.Create;
|
||||||
|
import org.apache.solr.client.solrj.request.CollectionAdminRequest.CreateAlias;
|
||||||
|
import org.apache.solr.client.solrj.request.CollectionAdminRequest.CreateShard;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Unit tests for {@link CollectionAdminRequest}.
|
||||||
|
*/
|
||||||
|
public class TestCollectionAdminRequest extends LuceneTestCase {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInvalidCollectionNameRejectedWhenCreatingCollection() {
|
||||||
|
final Create createRequest = new Create();
|
||||||
|
try {
|
||||||
|
createRequest.setCollectionName("invalid$collection@name");
|
||||||
|
fail();
|
||||||
|
} catch (IllegalArgumentException e) {
|
||||||
|
final String exceptionMessage = e.getMessage();
|
||||||
|
assertTrue(exceptionMessage.contains("Invalid collection"));
|
||||||
|
assertTrue(exceptionMessage.contains("invalid$collection@name"));
|
||||||
|
assertTrue(exceptionMessage.contains("must consist entirely of periods, underscores, and alphanumerics"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInvalidShardNamesRejectedWhenCreatingCollection() {
|
||||||
|
final Create createRequest = new Create();
|
||||||
|
try {
|
||||||
|
createRequest.setShards("invalid$shard@name");
|
||||||
|
fail();
|
||||||
|
} catch (IllegalArgumentException e) {
|
||||||
|
final String exceptionMessage = e.getMessage();
|
||||||
|
assertTrue(exceptionMessage.contains("Invalid shard"));
|
||||||
|
assertTrue(exceptionMessage.contains("invalid$shard@name"));
|
||||||
|
assertTrue(exceptionMessage.contains("must consist entirely of periods, underscores, and alphanumerics"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInvalidAliasNameRejectedWhenCreatingAlias() {
|
||||||
|
final CreateAlias createAliasRequest = new CreateAlias();
|
||||||
|
try {
|
||||||
|
createAliasRequest.setAliasName("invalid$alias@name");
|
||||||
|
fail();
|
||||||
|
} catch (IllegalArgumentException e) {
|
||||||
|
final String exceptionMessage = e.getMessage();
|
||||||
|
assertTrue(exceptionMessage.contains("Invalid collection"));
|
||||||
|
assertTrue(exceptionMessage.contains("invalid$alias@name"));
|
||||||
|
assertTrue(exceptionMessage.contains("must consist entirely of periods, underscores, and alphanumerics"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInvalidShardNameRejectedWhenCreatingShard() {
|
||||||
|
final CreateShard createShardRequest = new CreateShard();
|
||||||
|
try {
|
||||||
|
createShardRequest.setShardName("invalid$shard@name");
|
||||||
|
fail();
|
||||||
|
} catch (IllegalArgumentException e) {
|
||||||
|
final String exceptionMessage = e.getMessage();
|
||||||
|
assertTrue(exceptionMessage.contains("Invalid shard"));
|
||||||
|
assertTrue(exceptionMessage.contains("invalid$shard@name"));
|
||||||
|
assertTrue(exceptionMessage.contains("must consist entirely of periods, underscores, and alphanumerics"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -16,17 +16,19 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.client.solrj.request;
|
package org.apache.solr.client.solrj.request;
|
||||||
|
|
||||||
|
import static org.hamcrest.CoreMatchers.notNullValue;
|
||||||
|
import static org.hamcrest.core.Is.is;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.lang.invoke.MethodHandles;
|
import java.lang.invoke.MethodHandles;
|
||||||
|
|
||||||
import com.carrotsearch.randomizedtesting.annotations.ThreadLeakFilters;
|
|
||||||
import com.carrotsearch.randomizedtesting.rules.SystemPropertiesRestoreRule;
|
|
||||||
import org.apache.commons.io.FileUtils;
|
import org.apache.commons.io.FileUtils;
|
||||||
import org.apache.lucene.util.LuceneTestCase;
|
import org.apache.lucene.util.LuceneTestCase;
|
||||||
import org.apache.solr.SolrIgnoredThreadsFilter;
|
import org.apache.solr.SolrIgnoredThreadsFilter;
|
||||||
import org.apache.solr.client.solrj.SolrClient;
|
import org.apache.solr.client.solrj.SolrClient;
|
||||||
import org.apache.solr.client.solrj.embedded.AbstractEmbeddedSolrServerTestCase;
|
import org.apache.solr.client.solrj.embedded.AbstractEmbeddedSolrServerTestCase;
|
||||||
import org.apache.solr.client.solrj.embedded.EmbeddedSolrServer;
|
import org.apache.solr.client.solrj.embedded.EmbeddedSolrServer;
|
||||||
|
import org.apache.solr.client.solrj.request.CoreAdminRequest.Create;
|
||||||
import org.apache.solr.client.solrj.response.CoreAdminResponse;
|
import org.apache.solr.client.solrj.response.CoreAdminResponse;
|
||||||
import org.apache.solr.common.SolrException;
|
import org.apache.solr.common.SolrException;
|
||||||
import org.apache.solr.common.params.ModifiableSolrParams;
|
import org.apache.solr.common.params.ModifiableSolrParams;
|
||||||
|
@ -41,8 +43,8 @@ import org.junit.rules.TestRule;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
import static org.hamcrest.CoreMatchers.notNullValue;
|
import com.carrotsearch.randomizedtesting.annotations.ThreadLeakFilters;
|
||||||
import static org.hamcrest.core.Is.is;
|
import com.carrotsearch.randomizedtesting.rules.SystemPropertiesRestoreRule;
|
||||||
|
|
||||||
@ThreadLeakFilters(defaultFilters = true, filters = {SolrIgnoredThreadsFilter.class})
|
@ThreadLeakFilters(defaultFilters = true, filters = {SolrIgnoredThreadsFilter.class})
|
||||||
public class TestCoreAdmin extends AbstractEmbeddedSolrServerTestCase {
|
public class TestCoreAdmin extends AbstractEmbeddedSolrServerTestCase {
|
||||||
|
@ -158,6 +160,34 @@ public class TestCoreAdmin extends AbstractEmbeddedSolrServerTestCase {
|
||||||
assertTrue(gotExp);
|
assertTrue(gotExp);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInvalidCoreNamesAreRejectedWhenCreatingCore() {
|
||||||
|
final Create createRequest = new Create();
|
||||||
|
|
||||||
|
try {
|
||||||
|
createRequest.setCoreName("invalid$core@name");
|
||||||
|
fail();
|
||||||
|
} catch (IllegalArgumentException e) {
|
||||||
|
final String exceptionMessage = e.getMessage();
|
||||||
|
assertTrue(exceptionMessage.contains("Invalid core"));
|
||||||
|
assertTrue(exceptionMessage.contains("invalid$core@name"));
|
||||||
|
assertTrue(exceptionMessage.contains("must consist entirely of periods, underscores and alphanumerics"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testInvalidCoreNamesAreRejectedWhenRenamingExistingCore() throws Exception {
|
||||||
|
try {
|
||||||
|
CoreAdminRequest.renameCore("validExistingCoreName", "invalid$core@name", null);
|
||||||
|
fail();
|
||||||
|
} catch (IllegalArgumentException e) {
|
||||||
|
final String exceptionMessage = e.getMessage();
|
||||||
|
assertTrue(exceptionMessage.contains("Invalid core"));
|
||||||
|
assertTrue(exceptionMessage.contains("invalid$core@name"));
|
||||||
|
assertTrue(exceptionMessage.contains("must consist entirely of periods, underscores and alphanumerics"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@BeforeClass
|
@BeforeClass
|
||||||
public static void before() {
|
public static void before() {
|
||||||
// wtf?
|
// wtf?
|
||||||
|
|
Loading…
Reference in New Issue