mirror of https://github.com/apache/lucene.git
Merge branch 'apache-https-master' into jira/solr-8593
This commit is contained in:
commit
01f0ef4cbe
|
@ -382,6 +382,10 @@ Other Changes
|
||||||
|
|
||||||
* SOLR-9533: Reload core config when a core is reloaded (Gethin James, Joel Bernstein)
|
* SOLR-9533: Reload core config when a core is reloaded (Gethin James, Joel Bernstein)
|
||||||
|
|
||||||
|
* SOLR-9371: Fix bin/solr calculations for start/stop wait time and RMI_PORT.
|
||||||
|
(Shawn Heisey via Erick Erickson)
|
||||||
|
|
||||||
|
|
||||||
================== 6.2.1 ==================
|
================== 6.2.1 ==================
|
||||||
|
|
||||||
Bug Fixes
|
Bug Fixes
|
||||||
|
|
|
@ -119,6 +119,9 @@ else
|
||||||
JAVA=java
|
JAVA=java
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
if [ -z "$SOLR_STOP_WAIT" ]; then
|
||||||
|
SOLR_STOP_WAIT=180
|
||||||
|
fi
|
||||||
# test that Java exists, is executable and correct version
|
# test that Java exists, is executable and correct version
|
||||||
JAVA_VER=$("$JAVA" -version 2>&1)
|
JAVA_VER=$("$JAVA" -version 2>&1)
|
||||||
if [[ $? -ne 0 ]] ; then
|
if [[ $? -ne 0 ]] ; then
|
||||||
|
@ -231,7 +234,7 @@ function print_usage() {
|
||||||
echo ""
|
echo ""
|
||||||
echo " -p <port> Specify the port to start the Solr HTTP listener on; default is 8983"
|
echo " -p <port> Specify the port to start the Solr HTTP listener on; default is 8983"
|
||||||
echo " The specified port (SOLR_PORT) will also be used to determine the stop port"
|
echo " The specified port (SOLR_PORT) will also be used to determine the stop port"
|
||||||
echo " STOP_PORT=(\$SOLR_PORT-1000) and JMX RMI listen port RMI_PORT=(1\$SOLR_PORT). "
|
echo " STOP_PORT=(\$SOLR_PORT-1000) and JMX RMI listen port RMI_PORT=(\$SOLR_PORT+10000). "
|
||||||
echo " For instance, if you set -p 8985, then the STOP_PORT=7985 and RMI_PORT=18985"
|
echo " For instance, if you set -p 8985, then the STOP_PORT=7985 and RMI_PORT=18985"
|
||||||
echo ""
|
echo ""
|
||||||
echo " -d <dir> Specify the Solr server directory; defaults to server"
|
echo " -d <dir> Specify the Solr server directory; defaults to server"
|
||||||
|
@ -575,9 +578,24 @@ function stop_solr() {
|
||||||
SOLR_PID="$4"
|
SOLR_PID="$4"
|
||||||
|
|
||||||
if [ "$SOLR_PID" != "" ]; then
|
if [ "$SOLR_PID" != "" ]; then
|
||||||
echo -e "Sending stop command to Solr running on port $SOLR_PORT ... waiting 5 seconds to allow Jetty process $SOLR_PID to stop gracefully."
|
echo -e "Sending stop command to Solr running on port $SOLR_PORT ... waiting up to $SOLR_STOP_WAIT seconds to allow Jetty process $SOLR_PID to stop gracefully."
|
||||||
"$JAVA" $SOLR_SSL_OPTS $AUTHC_OPTS -jar "$DIR/start.jar" "STOP.PORT=$STOP_PORT" "STOP.KEY=$STOP_KEY" --stop || true
|
"$JAVA" $SOLR_SSL_OPTS $AUTHC_OPTS -jar "$DIR/start.jar" "STOP.PORT=$STOP_PORT" "STOP.KEY=$STOP_KEY" --stop || true
|
||||||
(sleep 5) &
|
(loops=0
|
||||||
|
while true
|
||||||
|
do
|
||||||
|
CHECK_PID=`ps auxww | awk '{print $2}' | grep -w $SOLR_PID | sort -r | tr -d ' '`
|
||||||
|
if [ "$CHECK_PID" != "" ]; then
|
||||||
|
slept=$((loops * 2))
|
||||||
|
if [ $slept -lt $SOLR_STOP_WAIT ]; then
|
||||||
|
sleep 2
|
||||||
|
loops=$[$loops+1]
|
||||||
|
else
|
||||||
|
exit # subshell!
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
exit # subshell!
|
||||||
|
fi
|
||||||
|
done) &
|
||||||
spinner $!
|
spinner $!
|
||||||
rm -f "$SOLR_PID_DIR/solr-$SOLR_PORT.pid"
|
rm -f "$SOLR_PID_DIR/solr-$SOLR_PORT.pid"
|
||||||
else
|
else
|
||||||
|
@ -1459,7 +1477,11 @@ fi
|
||||||
if [ "$ENABLE_REMOTE_JMX_OPTS" == "true" ]; then
|
if [ "$ENABLE_REMOTE_JMX_OPTS" == "true" ]; then
|
||||||
|
|
||||||
if [ -z "$RMI_PORT" ]; then
|
if [ -z "$RMI_PORT" ]; then
|
||||||
RMI_PORT="1$SOLR_PORT"
|
RMI_PORT=`expr $SOLR_PORT + 10000`
|
||||||
|
if [ $RMI_PORT -gt 65535 ]; then
|
||||||
|
echo -e "\nRMI_PORT is $RMI_PORT, which is invalid!\n"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
REMOTE_JMX_OPTS=('-Dcom.sun.management.jmxremote' \
|
REMOTE_JMX_OPTS=('-Dcom.sun.management.jmxremote' \
|
||||||
|
@ -1620,18 +1642,19 @@ function launch_solr() {
|
||||||
|
|
||||||
# no lsof on cygwin though
|
# no lsof on cygwin though
|
||||||
if hash lsof 2>/dev/null ; then # hash returns true if lsof is on the path
|
if hash lsof 2>/dev/null ; then # hash returns true if lsof is on the path
|
||||||
echo -n "Waiting up to 30 seconds to see Solr running on port $SOLR_PORT"
|
echo -n "Waiting up to $SOLR_STOP_WAIT seconds to see Solr running on port $SOLR_PORT"
|
||||||
# Launch in a subshell to show the spinner
|
# Launch in a subshell to show the spinner
|
||||||
(loops=0
|
(loops=0
|
||||||
while true
|
while true
|
||||||
do
|
do
|
||||||
running=`lsof -PniTCP:$SOLR_PORT -sTCP:LISTEN`
|
running=`lsof -PniTCP:$SOLR_PORT -sTCP:LISTEN`
|
||||||
if [ -z "$running" ]; then
|
if [ -z "$running" ]; then
|
||||||
if [ $loops -lt 6 ]; then
|
slept=$((loops * 2))
|
||||||
sleep 5
|
if [ $slept -lt $SOLR_STOP_WAIT ]; then
|
||||||
|
sleep 2
|
||||||
loops=$[$loops+1]
|
loops=$[$loops+1]
|
||||||
else
|
else
|
||||||
echo -e "Still not seeing Solr listening on $SOLR_PORT after 30 seconds!"
|
echo -e "Still not seeing Solr listening on $SOLR_PORT after $SOLR_STOP_WAIT seconds!"
|
||||||
tail -30 "$SOLR_LOGS_DIR/solr.log"
|
tail -30 "$SOLR_LOGS_DIR/solr.log"
|
||||||
exit # subshell!
|
exit # subshell!
|
||||||
fi
|
fi
|
||||||
|
|
|
@ -21,6 +21,12 @@
|
||||||
# affecting other Java applications on your server/workstation.
|
# affecting other Java applications on your server/workstation.
|
||||||
#SOLR_JAVA_HOME=""
|
#SOLR_JAVA_HOME=""
|
||||||
|
|
||||||
|
# This controls the number of seconds that the solr script will wait for
|
||||||
|
# Solr to stop gracefully or Solr to start. If the graceful stop fails,
|
||||||
|
# the script will forcibly stop Solr. If the start fails, the script will
|
||||||
|
# give up waiting and display the last few lines of the logfile.
|
||||||
|
#SOLR_STOP_WAIT="180"
|
||||||
|
|
||||||
# Increase Java Heap as needed to support your indexing / query needs
|
# Increase Java Heap as needed to support your indexing / query needs
|
||||||
#SOLR_HEAP="512m"
|
#SOLR_HEAP="512m"
|
||||||
|
|
||||||
|
|
|
@ -38,6 +38,8 @@ import java.util.Random;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
import java.util.concurrent.atomic.AtomicLong;
|
import java.util.concurrent.atomic.AtomicLong;
|
||||||
|
|
||||||
|
import org.apache.solr.client.solrj.SolrClient;
|
||||||
|
import org.apache.solr.client.solrj.impl.HttpSolrClient;
|
||||||
import org.apache.solr.core.CoreContainer;
|
import org.apache.solr.core.CoreContainer;
|
||||||
import org.apache.solr.servlet.SolrDispatchFilter;
|
import org.apache.solr.servlet.SolrDispatchFilter;
|
||||||
import org.eclipse.jetty.server.Connector;
|
import org.eclipse.jetty.server.Connector;
|
||||||
|
@ -293,6 +295,10 @@ public class JettySolrRunner {
|
||||||
return getSolrDispatchFilter().getCores();
|
return getSolrDispatchFilter().getCores();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public String getNodeName() {
|
||||||
|
return getCoreContainer().getZkController().getNodeName();
|
||||||
|
}
|
||||||
|
|
||||||
public boolean isRunning() {
|
public boolean isRunning() {
|
||||||
return server.isRunning();
|
return server.isRunning();
|
||||||
}
|
}
|
||||||
|
@ -453,6 +459,10 @@ public class JettySolrRunner {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public SolrClient newClient() {
|
||||||
|
return new HttpSolrClient.Builder(getBaseUrl().toString()).build();
|
||||||
|
}
|
||||||
|
|
||||||
public DebugFilter getDebugFilter() {
|
public DebugFilter getDebugFilter() {
|
||||||
return (DebugFilter)debugFilter.getFilter();
|
return (DebugFilter)debugFilter.getFilter();
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,34 +0,0 @@
|
||||||
/*
|
|
||||||
* 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.cloud.rule;
|
|
||||||
|
|
||||||
import java.util.Set;
|
|
||||||
|
|
||||||
import com.google.common.collect.ImmutableSet;
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
public abstract class Snitch {
|
|
||||||
static Set<Class> WELL_KNOWN_SNITCHES = ImmutableSet.of(ImplicitSnitch.class);
|
|
||||||
|
|
||||||
|
|
||||||
public abstract void getTags(String solrNode, Set<String> requestedTags, org.apache.solr.common.cloud.rule.SnitchContext ctx);
|
|
||||||
|
|
||||||
public abstract boolean isKnownTag(String tag);
|
|
||||||
|
|
||||||
}
|
|
|
@ -3262,7 +3262,7 @@ public class SolrCLI {
|
||||||
.create("m"),
|
.create("m"),
|
||||||
OptionBuilder
|
OptionBuilder
|
||||||
.withDescription("Timeout in ms for commands supporting a timeout")
|
.withDescription("Timeout in ms for commands supporting a timeout")
|
||||||
.withLongOpt("ms")
|
.withLongOpt("timeout")
|
||||||
.hasArg(true)
|
.hasArg(true)
|
||||||
.withType(Long.class)
|
.withType(Long.class)
|
||||||
.withArgName("ms")
|
.withArgName("ms")
|
||||||
|
|
|
@ -223,6 +223,8 @@
|
||||||
<!-- points to the root document of a block of nested documents -->
|
<!-- points to the root document of a block of nested documents -->
|
||||||
<field name="_root_" type="string" indexed="true" stored="true"/>
|
<field name="_root_" type="string" indexed="true" stored="true"/>
|
||||||
|
|
||||||
|
<field name="_route_" type="string" indexed="true" stored="true" multiValued="false"/>
|
||||||
|
|
||||||
<field name="multi_int_with_docvals" type="tint" multiValued="true" docValues="true" indexed="false"/>
|
<field name="multi_int_with_docvals" type="tint" multiValued="true" docValues="true" indexed="false"/>
|
||||||
|
|
||||||
<dynamicField name="*_coordinate" type="tdouble" indexed="true" stored="false"/>
|
<dynamicField name="*_coordinate" type="tdouble" indexed="true" stored="false"/>
|
||||||
|
|
|
@ -0,0 +1,28 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8" ?>
|
||||||
|
<!--
|
||||||
|
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.
|
||||||
|
-->
|
||||||
|
<schema name="minimal" version="1.1">
|
||||||
|
<fieldType name="string" class="solr.StrField"/>
|
||||||
|
<fieldType name="int" class="solr.TrieIntField" precisionStep="0" omitNorms="true" positionIncrementGap="0"/>
|
||||||
|
<fieldType name="long" class="solr.TrieLongField" precisionStep="0" omitNorms="true" positionIncrementGap="0"/>
|
||||||
|
<dynamicField name="*" type="string" indexed="true" stored="true"/>
|
||||||
|
<!-- for versioning -->
|
||||||
|
<field name="_version_" type="long" indexed="true" stored="true"/>
|
||||||
|
<field name="_root_" type="int" indexed="true" stored="true" multiValued="false" required="false"/>
|
||||||
|
<field name="id" type="string" indexed="true" stored="true"/>
|
||||||
|
<uniqueKey>id</uniqueKey>
|
||||||
|
</schema>
|
|
@ -0,0 +1,50 @@
|
||||||
|
<?xml version="1.0" ?>
|
||||||
|
|
||||||
|
<!--
|
||||||
|
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.
|
||||||
|
-->
|
||||||
|
|
||||||
|
<!-- Minimal solrconfig.xml with /select, /admin and /update only -->
|
||||||
|
|
||||||
|
<config>
|
||||||
|
|
||||||
|
<directoryFactory name="DirectoryFactory"
|
||||||
|
class="solr.HdfsDirectoryFactory"/>
|
||||||
|
<indexConfig>
|
||||||
|
<lockType>hdfs</lockType>
|
||||||
|
</indexConfig>
|
||||||
|
|
||||||
|
<schemaFactory class="ClassicIndexSchemaFactory"/>
|
||||||
|
|
||||||
|
<luceneMatchVersion>${tests.luceneMatchVersion:LATEST}</luceneMatchVersion>
|
||||||
|
|
||||||
|
<updateHandler class="solr.DirectUpdateHandler2">
|
||||||
|
<commitWithin>
|
||||||
|
<softCommit>${solr.commitwithin.softcommit:true}</softCommit>
|
||||||
|
</commitWithin>
|
||||||
|
<updateLog></updateLog>
|
||||||
|
</updateHandler>
|
||||||
|
|
||||||
|
<requestHandler name="/select" class="solr.SearchHandler">
|
||||||
|
<lst name="defaults">
|
||||||
|
<str name="echoParams">explicit</str>
|
||||||
|
<str name="indent">true</str>
|
||||||
|
<str name="df">text</str>
|
||||||
|
</lst>
|
||||||
|
|
||||||
|
</requestHandler>
|
||||||
|
</config>
|
||||||
|
|
|
@ -0,0 +1,28 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8" ?>
|
||||||
|
<!--
|
||||||
|
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.
|
||||||
|
-->
|
||||||
|
<schema name="minimal" version="1.1">
|
||||||
|
<fieldType name="string" class="solr.StrField"/>
|
||||||
|
<fieldType name="int" class="solr.TrieIntField" precisionStep="0" omitNorms="true" positionIncrementGap="0"/>
|
||||||
|
<fieldType name="long" class="solr.TrieLongField" precisionStep="0" omitNorms="true" positionIncrementGap="0"/>
|
||||||
|
<dynamicField name="*" type="string" indexed="true" stored="true"/>
|
||||||
|
<!-- for versioning -->
|
||||||
|
<field name="_version_" type="long" indexed="true" stored="true"/>
|
||||||
|
<field name="_root_" type="int" indexed="true" stored="true" multiValued="false" required="false"/>
|
||||||
|
<field name="id" type="string" indexed="true" stored="true"/>
|
||||||
|
<uniqueKey>id</uniqueKey>
|
||||||
|
</schema>
|
|
@ -0,0 +1,50 @@
|
||||||
|
<?xml version="1.0" ?>
|
||||||
|
|
||||||
|
<!--
|
||||||
|
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.
|
||||||
|
-->
|
||||||
|
|
||||||
|
<!-- Minimal solrconfig.xml with JMX enabled -->
|
||||||
|
|
||||||
|
<config>
|
||||||
|
|
||||||
|
<jmx/>
|
||||||
|
|
||||||
|
<dataDir>${solr.data.dir:}</dataDir>
|
||||||
|
|
||||||
|
<directoryFactory name="DirectoryFactory"
|
||||||
|
class="${solr.directoryFactory:solr.NRTCachingDirectoryFactory}"/>
|
||||||
|
<schemaFactory class="ClassicIndexSchemaFactory"/>
|
||||||
|
|
||||||
|
<luceneMatchVersion>${tests.luceneMatchVersion:LATEST}</luceneMatchVersion>
|
||||||
|
|
||||||
|
<updateHandler class="solr.DirectUpdateHandler2">
|
||||||
|
<commitWithin>
|
||||||
|
<softCommit>${solr.commitwithin.softcommit:true}</softCommit>
|
||||||
|
</commitWithin>
|
||||||
|
<updateLog></updateLog>
|
||||||
|
</updateHandler>
|
||||||
|
|
||||||
|
<requestHandler name="/select" class="solr.SearchHandler">
|
||||||
|
<lst name="defaults">
|
||||||
|
<str name="echoParams">explicit</str>
|
||||||
|
<str name="indent">true</str>
|
||||||
|
<str name="df">text</str>
|
||||||
|
</lst>
|
||||||
|
|
||||||
|
</requestHandler>
|
||||||
|
</config>
|
||||||
|
|
|
@ -16,186 +16,153 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.cloud;
|
package org.apache.solr.cloud;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Properties;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
import org.apache.commons.lang.StringUtils;
|
import org.apache.commons.lang.StringUtils;
|
||||||
import org.apache.lucene.util.LuceneTestCase.Slow;
|
import org.apache.lucene.util.LuceneTestCase.Slow;
|
||||||
import org.apache.solr.client.solrj.embedded.JettySolrRunner;
|
import org.apache.solr.client.solrj.embedded.JettySolrRunner;
|
||||||
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.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.cloud.ZkStateReader;
|
|
||||||
import org.apache.solr.common.util.NamedList;
|
|
||||||
import org.apache.zookeeper.KeeperException;
|
import org.apache.zookeeper.KeeperException;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
@Slow
|
@Slow
|
||||||
public class CollectionTooManyReplicasTest extends AbstractFullDistribZkTestBase {
|
public class CollectionTooManyReplicasTest extends SolrCloudTestCase {
|
||||||
|
|
||||||
public CollectionTooManyReplicasTest() {
|
@BeforeClass
|
||||||
sliceCount = 1;
|
public static void setupCluster() throws Exception {
|
||||||
|
configureCluster(3)
|
||||||
|
.addConfig("conf", configset("cloud-minimal"))
|
||||||
|
.configure();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void deleteCollections() throws Exception {
|
||||||
|
cluster.deleteAllCollections();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ShardsFixed(num = 1)
|
|
||||||
public void testAddTooManyReplicas() throws Exception {
|
public void testAddTooManyReplicas() throws Exception {
|
||||||
String collectionName = "TooManyReplicasInSeveralFlavors";
|
final String collectionName = "TooManyReplicasInSeveralFlavors";
|
||||||
CollectionAdminRequest.Create create = new CollectionAdminRequest.Create()
|
CollectionAdminRequest.createCollection(collectionName, "conf", 2, 1)
|
||||||
.setCollectionName(collectionName)
|
.setMaxShardsPerNode(1)
|
||||||
.setNumShards(2)
|
.process(cluster.getSolrClient());
|
||||||
.setReplicationFactor(1)
|
|
||||||
.setMaxShardsPerNode(2)
|
|
||||||
.setStateFormat(2);
|
|
||||||
|
|
||||||
CollectionAdminResponse response = create.process(cloudClient);
|
// I have two replicas, one for each shard
|
||||||
assertEquals(0, response.getStatus());
|
|
||||||
assertTrue(response.isSuccess());
|
|
||||||
// Now I have the fixed Jetty plus the control instnace, I have two replicas, one for each shard
|
|
||||||
|
|
||||||
// Curiously, I should be able to add a bunch of replicas if I specify the node, even more than maxShardsPerNode
|
// Curiously, I should be able to add a bunch of replicas if I specify the node, even more than maxShardsPerNode
|
||||||
// Just get the first node any way we can.
|
// Just get the first node any way we can.
|
||||||
// Get a node to use for the "node" parameter.
|
// Get a node to use for the "node" parameter.
|
||||||
|
|
||||||
String nodeName = getAllNodeNames(collectionName).get(0);
|
String nodeName = getAllNodeNames(collectionName).get(0);
|
||||||
|
|
||||||
// Add a replica using the "node" parameter (no "too many replicas check")
|
// Add a replica using the "node" parameter (no "too many replicas check")
|
||||||
// this node should have 2 replicas on it
|
// this node should have 2 replicas on it
|
||||||
CollectionAdminRequest.AddReplica addReplicaNode = new CollectionAdminRequest.AddReplica()
|
CollectionAdminRequest.addReplicaToShard(collectionName, "shard1")
|
||||||
.setCollectionName(collectionName)
|
.setNode(nodeName)
|
||||||
.setShardName("shard1")
|
.process(cluster.getSolrClient());
|
||||||
.setNode(nodeName);
|
|
||||||
response = addReplicaNode.process(cloudClient);
|
|
||||||
assertEquals(0, response.getStatus());
|
|
||||||
|
|
||||||
// Three replicas so far, should be able to create another one "normally"
|
// Three replicas so far, should be able to create another one "normally"
|
||||||
CollectionAdminRequest.AddReplica addReplica = new CollectionAdminRequest.AddReplica()
|
CollectionAdminRequest.addReplicaToShard(collectionName, "shard1")
|
||||||
.setCollectionName(collectionName)
|
.process(cluster.getSolrClient());
|
||||||
.setShardName("shard1");
|
|
||||||
|
|
||||||
response = addReplica.process(cloudClient);
|
|
||||||
assertEquals(0, response.getStatus());
|
|
||||||
|
|
||||||
// This one should fail though, no "node" parameter specified
|
// This one should fail though, no "node" parameter specified
|
||||||
try {
|
Exception e = expectThrows(Exception.class, () -> {
|
||||||
addReplica.process(cloudClient);
|
CollectionAdminRequest.addReplicaToShard(collectionName, "shard1")
|
||||||
fail("Should have thrown an error because the nodes are full");
|
.process(cluster.getSolrClient());
|
||||||
} catch (HttpSolrClient.RemoteSolrException se) {
|
});
|
||||||
|
|
||||||
assertTrue("Should have gotten the right error message back",
|
assertTrue("Should have gotten the right error message back",
|
||||||
se.getMessage().contains("given the current number of live nodes and a maxShardsPerNode of"));
|
e.getMessage().contains("given the current number of live nodes and a maxShardsPerNode of"));
|
||||||
}
|
|
||||||
|
|
||||||
// Oddly, we should succeed next just because setting property.name will not check for nodes being "full up"
|
// Oddly, we should succeed next just because setting property.name will not check for nodes being "full up"
|
||||||
Properties props = new Properties();
|
// TODO: Isn't this a bug?
|
||||||
props.setProperty("name", "bogus2");
|
CollectionAdminRequest.addReplicaToShard(collectionName, "shard1")
|
||||||
addReplicaNode.setProperties(props);
|
.withProperty("name", "bogus2")
|
||||||
response = addReplicaNode.process(cloudClient);
|
.setNode(nodeName)
|
||||||
assertEquals(0, response.getStatus());
|
.process(cluster.getSolrClient());
|
||||||
|
|
||||||
ZkStateReader zkStateReader = getCommonCloudSolrClient().getZkStateReader();
|
DocCollection collectionState = getCollectionState(collectionName);
|
||||||
zkStateReader.forceUpdateCollection(collectionName);
|
Slice slice = collectionState.getSlice("shard1");
|
||||||
Slice slice = zkStateReader.getClusterState().getSlicesMap(collectionName).get("shard1");
|
Replica replica = getRandomReplica(slice, r -> r.getCoreName().equals("bogus2"));
|
||||||
|
assertNotNull("Should have found a replica named 'bogus2'", replica);
|
||||||
Replica rep = null;
|
assertEquals("Replica should have been put on correct core", nodeName, replica.getNodeName());
|
||||||
for (Replica rep1 : slice.getReplicas()) { // Silly compiler
|
|
||||||
if (rep1.get("core").equals("bogus2")) {
|
|
||||||
rep = rep1;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
assertNotNull("Should have found a replica named 'bogus2'", rep);
|
|
||||||
assertEquals("Replica should have been put on correct core", nodeName, rep.getNodeName());
|
|
||||||
|
|
||||||
// Shard1 should have 4 replicas
|
// Shard1 should have 4 replicas
|
||||||
assertEquals("There should be 4 replicas for shard 1", 4, slice.getReplicas().size());
|
assertEquals("There should be 4 replicas for shard 1", 4, slice.getReplicas().size());
|
||||||
|
|
||||||
// And let's fail one more time because to insure that the math doesn't do weird stuff it we have more replicas
|
// And let's fail one more time because to ensure that the math doesn't do weird stuff it we have more replicas
|
||||||
// than simple calcs would indicate.
|
// than simple calcs would indicate.
|
||||||
try {
|
Exception e2 = expectThrows(Exception.class, () -> {
|
||||||
addReplica.process(cloudClient);
|
CollectionAdminRequest.addReplicaToShard(collectionName, "shard1")
|
||||||
fail("Should have thrown an error because the nodes are full");
|
.process(cluster.getSolrClient());
|
||||||
} catch (HttpSolrClient.RemoteSolrException se) {
|
});
|
||||||
|
|
||||||
assertTrue("Should have gotten the right error message back",
|
assertTrue("Should have gotten the right error message back",
|
||||||
se.getMessage().contains("given the current number of live nodes and a maxShardsPerNode of"));
|
e2.getMessage().contains("given the current number of live nodes and a maxShardsPerNode of"));
|
||||||
|
|
||||||
|
// wait for recoveries to finish, for a clean shutdown - see SOLR-9645
|
||||||
|
waitForState("Expected to see all replicas active", collectionName, (n, c) -> {
|
||||||
|
for (Replica r : c.getReplicas()) {
|
||||||
|
if (r.getState() != Replica.State.ACTIVE)
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
return true;
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ShardsFixed(num = 2)
|
|
||||||
public void testAddShard() throws Exception {
|
public void testAddShard() throws Exception {
|
||||||
|
|
||||||
String collectionName = "TooManyReplicasWhenAddingShards";
|
String collectionName = "TooManyReplicasWhenAddingShards";
|
||||||
CollectionAdminRequest.Create create = new CollectionAdminRequest.Create()
|
CollectionAdminRequest.createCollectionWithImplicitRouter(collectionName, "conf", "shardstart", 2)
|
||||||
.setCollectionName(collectionName)
|
|
||||||
.setReplicationFactor(2)
|
|
||||||
.setMaxShardsPerNode(2)
|
.setMaxShardsPerNode(2)
|
||||||
.setStateFormat(2)
|
.process(cluster.getSolrClient());
|
||||||
.setRouterName("implicit")
|
|
||||||
.setShards("shardstart");
|
|
||||||
|
|
||||||
NamedList<Object> request = create.process(cloudClient).getResponse();
|
|
||||||
|
|
||||||
assertTrue("Could not create the collection", request.get("success") != null);
|
|
||||||
// We have two nodes, maxShardsPerNode is set to 2. Therefore, we should be able to add 2 shards each with
|
// We have two nodes, maxShardsPerNode is set to 2. Therefore, we should be able to add 2 shards each with
|
||||||
// two replicas, but fail on the third.
|
// two replicas, but fail on the third.
|
||||||
|
CollectionAdminRequest.createShard(collectionName, "shard1")
|
||||||
CollectionAdminRequest.CreateShard createShard = new CollectionAdminRequest.CreateShard()
|
.process(cluster.getSolrClient());
|
||||||
.setCollectionName(collectionName)
|
|
||||||
.setShardName("shard1");
|
|
||||||
CollectionAdminResponse resp = createShard.process(cloudClient);
|
|
||||||
assertEquals(0, resp.getStatus());
|
|
||||||
|
|
||||||
// Now we should have one replica on each Jetty, add another to reach maxShardsPerNode
|
// Now we should have one replica on each Jetty, add another to reach maxShardsPerNode
|
||||||
|
CollectionAdminRequest.createShard(collectionName, "shard2")
|
||||||
createShard = new CollectionAdminRequest.CreateShard()
|
.process(cluster.getSolrClient());
|
||||||
.setCollectionName(collectionName)
|
|
||||||
.setShardName("shard2");
|
|
||||||
resp = createShard.process(cloudClient);
|
|
||||||
assertEquals(0, resp.getStatus());
|
|
||||||
|
|
||||||
|
|
||||||
// Now fail to add the third as it should exceed maxShardsPerNode
|
// Now fail to add the third as it should exceed maxShardsPerNode
|
||||||
createShard = new CollectionAdminRequest.CreateShard()
|
Exception e = expectThrows(Exception.class, () -> {
|
||||||
.setCollectionName(collectionName)
|
CollectionAdminRequest.createShard(collectionName, "shard3")
|
||||||
.setShardName("shard3");
|
.process(cluster.getSolrClient());
|
||||||
try {
|
});
|
||||||
createShard.process(cloudClient);
|
|
||||||
fail("Should have exceeded the max number of replicas allowed");
|
|
||||||
} catch (HttpSolrClient.RemoteSolrException se) {
|
|
||||||
assertTrue("Should have gotten the right error message back",
|
assertTrue("Should have gotten the right error message back",
|
||||||
se.getMessage().contains("given the current number of live nodes and a maxShardsPerNode of"));
|
e.getMessage().contains("given the current number of live nodes and a maxShardsPerNode of"));
|
||||||
}
|
|
||||||
|
|
||||||
// Hmmm, providing a nodeset also overrides the checks for max replicas, so prove it.
|
// Hmmm, providing a nodeset also overrides the checks for max replicas, so prove it.
|
||||||
List<String> nodes = getAllNodeNames(collectionName);
|
List<String> nodes = getAllNodeNames(collectionName);
|
||||||
|
|
||||||
createShard = new CollectionAdminRequest.CreateShard()
|
CollectionAdminRequest.createShard(collectionName, "shard4")
|
||||||
.setCollectionName(collectionName)
|
.setNodeSet(StringUtils.join(nodes, ","))
|
||||||
.setShardName("shard4")
|
.process(cluster.getSolrClient());
|
||||||
.setNodeSet(StringUtils.join(nodes, ","));
|
|
||||||
resp = createShard.process(cloudClient);
|
|
||||||
assertEquals(0, resp.getStatus());
|
|
||||||
|
|
||||||
// And just for yucks, insure we fail the "regular" one again.
|
// And just for yucks, insure we fail the "regular" one again.
|
||||||
createShard = new CollectionAdminRequest.CreateShard()
|
Exception e2 = expectThrows(Exception.class, () -> {
|
||||||
.setCollectionName(collectionName)
|
CollectionAdminRequest.createShard(collectionName, "shard5")
|
||||||
.setShardName("shard5");
|
.process(cluster.getSolrClient());
|
||||||
try {
|
});
|
||||||
createShard.process(cloudClient);
|
|
||||||
fail("Should have exceeded the max number of replicas allowed");
|
|
||||||
} catch (HttpSolrClient.RemoteSolrException se) {
|
|
||||||
assertTrue("Should have gotten the right error message back",
|
assertTrue("Should have gotten the right error message back",
|
||||||
se.getMessage().contains("given the current number of live nodes and a maxShardsPerNode of"));
|
e2.getMessage().contains("given the current number of live nodes and a maxShardsPerNode of"));
|
||||||
}
|
|
||||||
|
|
||||||
// And finally, insure that there are all the replcias we expect. We should have shards 1, 2 and 4 and each
|
// And finally, insure that there are all the replcias we expect. We should have shards 1, 2 and 4 and each
|
||||||
// should have exactly two replicas
|
// should have exactly two replicas
|
||||||
ZkStateReader zkStateReader = getCommonCloudSolrClient().getZkStateReader();
|
waitForState("Expected shards shardstart, 1, 2 and 4, each with two active replicas", collectionName, (n, c) -> {
|
||||||
zkStateReader.forceUpdateCollection(collectionName);
|
return DocCollection.isFullyActive(n, c, 4, 2);
|
||||||
Map<String, Slice> slices = zkStateReader.getClusterState().getSlicesMap(collectionName);
|
});
|
||||||
|
Map<String, Slice> slices = getCollectionState(collectionName).getSlicesMap();
|
||||||
assertEquals("There should be exaclty four slices", slices.size(), 4);
|
assertEquals("There should be exaclty four slices", slices.size(), 4);
|
||||||
assertNotNull("shardstart should exist", slices.get("shardstart"));
|
assertNotNull("shardstart should exist", slices.get("shardstart"));
|
||||||
assertNotNull("shard1 should exist", slices.get("shard1"));
|
assertNotNull("shard1 should exist", slices.get("shard1"));
|
||||||
|
@ -209,82 +176,46 @@ public class CollectionTooManyReplicasTest extends AbstractFullDistribZkTestBase
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ShardsFixed(num = 2)
|
|
||||||
public void testDownedShards() throws Exception {
|
public void testDownedShards() throws Exception {
|
||||||
String collectionName = "TooManyReplicasWhenAddingDownedNode";
|
String collectionName = "TooManyReplicasWhenAddingDownedNode";
|
||||||
CollectionAdminRequest.Create create = new CollectionAdminRequest.Create()
|
CollectionAdminRequest.createCollectionWithImplicitRouter(collectionName, "conf", "shardstart", 1)
|
||||||
.setCollectionName(collectionName)
|
|
||||||
.setReplicationFactor(1)
|
|
||||||
.setMaxShardsPerNode(2)
|
.setMaxShardsPerNode(2)
|
||||||
.setStateFormat(2)
|
.process(cluster.getSolrClient());
|
||||||
.setRouterName("implicit")
|
|
||||||
.setShards("shardstart");
|
|
||||||
|
|
||||||
NamedList<Object> request = create.process(cloudClient).getResponse();
|
|
||||||
|
|
||||||
assertTrue("Could not create the collection", request.get("success") != null);
|
|
||||||
try (SolrZkClient zkClient = new SolrZkClient(zkServer.getZkAddress(),
|
|
||||||
AbstractZkTestCase.TIMEOUT)) {
|
|
||||||
|
|
||||||
List<String> liveNodes = zkClient.getChildren("/live_nodes", null, true);
|
|
||||||
|
|
||||||
// Shut down a Jetty, I really don't care which
|
// Shut down a Jetty, I really don't care which
|
||||||
JettySolrRunner downJetty = jettys.get(r.nextInt(2));
|
JettySolrRunner jetty = cluster.getRandomJetty(random());
|
||||||
|
String deadNode = jetty.getBaseUrl().toString();
|
||||||
downJetty.stop();
|
cluster.stopJettySolrRunner(jetty);
|
||||||
List<String> liveNodesNow = null;
|
|
||||||
for (int idx = 0; idx < 150; ++idx) {
|
|
||||||
liveNodesNow = zkClient.getChildren("/live_nodes", null, true);
|
|
||||||
if (liveNodesNow.size() != liveNodes.size()) break;
|
|
||||||
Thread.sleep(100);
|
|
||||||
}
|
|
||||||
List<String> deadNodes = new ArrayList<>(liveNodes);
|
|
||||||
assertTrue("Should be a downed node", deadNodes.removeAll(liveNodesNow));
|
|
||||||
liveNodes.removeAll(deadNodes);
|
|
||||||
|
|
||||||
//OK, we've killed a node. Insure we get errors when we ask to create a replica or shard that involves it.
|
|
||||||
// First try adding a replica to the downed node.
|
|
||||||
CollectionAdminRequest.AddReplica addReplicaNode = new CollectionAdminRequest.AddReplica()
|
|
||||||
.setCollectionName(collectionName)
|
|
||||||
.setShardName("shardstart")
|
|
||||||
.setNode(deadNodes.get(0));
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
addReplicaNode.process(cloudClient);
|
|
||||||
fail("Should have gotten an exception");
|
// Adding a replica on a dead node should fail
|
||||||
} catch (HttpSolrClient.RemoteSolrException se) {
|
Exception e1 = expectThrows(Exception.class, () -> {
|
||||||
|
CollectionAdminRequest.addReplicaToShard(collectionName, "shardstart")
|
||||||
|
.setNode(deadNode)
|
||||||
|
.process(cluster.getSolrClient());
|
||||||
|
});
|
||||||
assertTrue("Should have gotten a message about shard not ",
|
assertTrue("Should have gotten a message about shard not ",
|
||||||
se.getMessage().contains("At least one of the node(s) specified are not currently active, no action taken."));
|
e1.getMessage().contains("At least one of the node(s) specified are not currently active, no action taken."));
|
||||||
}
|
|
||||||
|
|
||||||
// Should also die if we just add a shard
|
// Should also die if we just add a shard
|
||||||
CollectionAdminRequest.CreateShard createShard = new CollectionAdminRequest.CreateShard()
|
Exception e2 = expectThrows(Exception.class, () -> {
|
||||||
.setCollectionName(collectionName)
|
CollectionAdminRequest.createShard(collectionName, "shard1")
|
||||||
.setShardName("shard1")
|
.setNodeSet(deadNode)
|
||||||
.setNodeSet(deadNodes.get(0));
|
.process(cluster.getSolrClient());
|
||||||
try {
|
});
|
||||||
createShard.process(cloudClient);
|
|
||||||
fail("Should have gotten an exception");
|
|
||||||
} catch (HttpSolrClient.RemoteSolrException se) {
|
|
||||||
assertTrue("Should have gotten a message about shard not ",
|
assertTrue("Should have gotten a message about shard not ",
|
||||||
se.getMessage().contains("At least one of the node(s) specified are not currently active, no action taken."));
|
e2.getMessage().contains("At least one of the node(s) specified are not currently active, no action taken."));
|
||||||
}
|
}
|
||||||
//downJetty.start();
|
finally {
|
||||||
|
cluster.startJettySolrRunner(jetty);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private List<String> getAllNodeNames(String collectionName) throws KeeperException, InterruptedException {
|
private List<String> getAllNodeNames(String collectionName) throws KeeperException, InterruptedException {
|
||||||
ZkStateReader zkStateReader = getCommonCloudSolrClient().getZkStateReader();
|
DocCollection state = getCollectionState(collectionName);
|
||||||
zkStateReader.forceUpdateCollection(collectionName);
|
return state.getReplicas().stream().map(Replica::getNodeName).distinct().collect(Collectors.toList());
|
||||||
Slice slice = zkStateReader.getClusterState().getSlicesMap(collectionName).get("shard1");
|
|
||||||
|
|
||||||
List<String> nodes = new ArrayList<>();
|
|
||||||
for (Replica rep : slice.getReplicas()) {
|
|
||||||
nodes.add(rep.getNodeName());
|
|
||||||
}
|
|
||||||
|
|
||||||
assertTrue("Should have some nodes!", nodes.size() > 0);
|
|
||||||
return nodes;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -17,7 +17,6 @@
|
||||||
|
|
||||||
package org.apache.solr.cloud;
|
package org.apache.solr.cloud;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Properties;
|
import java.util.Properties;
|
||||||
|
|
||||||
import org.apache.solr.client.solrj.impl.CloudSolrClient;
|
import org.apache.solr.client.solrj.impl.CloudSolrClient;
|
||||||
|
@ -77,8 +76,7 @@ public class CreateCollectionCleanupTest extends SolrCloudTestCase {
|
||||||
assertFalse(rsp.isSuccess());
|
assertFalse(rsp.isSuccess());
|
||||||
|
|
||||||
// Confirm using LIST that the collection does not exist
|
// Confirm using LIST that the collection does not exist
|
||||||
CollectionAdminRequest.List list = CollectionAdminRequest.listCollections();
|
assertFalse(CollectionAdminRequest.listCollections(cloudClient).contains("foo"));
|
||||||
rsp = list.process(cloudClient);
|
|
||||||
assertFalse(((ArrayList) rsp.getResponse().get("collections")).contains("foo"));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,41 +16,19 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.cloud;
|
package org.apache.solr.cloud;
|
||||||
|
|
||||||
import org.apache.lucene.util.LuceneTestCase.Slow;
|
|
||||||
import org.apache.lucene.util.TestUtil;
|
|
||||||
import org.apache.solr.SolrTestCaseJ4.SuppressSSL;
|
|
||||||
import org.apache.solr.client.solrj.SolrClient;
|
|
||||||
import org.apache.solr.client.solrj.SolrQuery;
|
|
||||||
import org.apache.solr.client.solrj.SolrRequest;
|
|
||||||
import org.apache.solr.client.solrj.SolrServerException;
|
|
||||||
import org.apache.solr.client.solrj.impl.CloudSolrClient;
|
|
||||||
import org.apache.solr.client.solrj.impl.HttpSolrClient;
|
|
||||||
import org.apache.solr.client.solrj.request.QueryRequest;
|
|
||||||
import org.apache.solr.client.solrj.request.UpdateRequest;
|
|
||||||
import org.apache.solr.client.solrj.response.QueryResponse;
|
|
||||||
import org.apache.solr.common.cloud.ClusterState;
|
|
||||||
import org.apache.solr.common.cloud.DocCollection;
|
|
||||||
import org.apache.solr.common.cloud.ImplicitDocRouter;
|
|
||||||
import org.apache.solr.common.cloud.Replica;
|
|
||||||
import org.apache.solr.common.cloud.ZkStateReader;
|
|
||||||
import org.apache.solr.common.params.CollectionParams.CollectionAction;
|
|
||||||
import org.apache.solr.common.params.ModifiableSolrParams;
|
|
||||||
import org.apache.solr.common.util.Utils;
|
|
||||||
import org.junit.Test;
|
|
||||||
import org.slf4j.Logger;
|
|
||||||
import org.slf4j.LoggerFactory;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.lang.invoke.MethodHandles;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Map.Entry;
|
|
||||||
import java.util.Set;
|
|
||||||
|
|
||||||
import static org.apache.solr.cloud.OverseerCollectionMessageHandler.NUM_SLICES;
|
import org.apache.lucene.util.TestUtil;
|
||||||
import static org.apache.solr.cloud.OverseerCollectionMessageHandler.SHARDS_PROP;
|
import org.apache.solr.client.solrj.SolrQuery;
|
||||||
|
import org.apache.solr.client.solrj.request.CollectionAdminRequest;
|
||||||
|
import org.apache.solr.client.solrj.request.UpdateRequest;
|
||||||
|
import org.apache.solr.common.SolrInputDocument;
|
||||||
|
import org.apache.solr.common.cloud.DocCollection;
|
||||||
|
import org.apache.solr.common.cloud.Replica;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
import static org.apache.solr.common.cloud.DocCollection.DOC_ROUTER;
|
import static org.apache.solr.common.cloud.DocCollection.DOC_ROUTER;
|
||||||
import static org.apache.solr.common.cloud.ZkStateReader.MAX_SHARDS_PER_NODE;
|
import static org.apache.solr.common.cloud.ZkStateReader.MAX_SHARDS_PER_NODE;
|
||||||
import static org.apache.solr.common.cloud.ZkStateReader.REPLICATION_FACTOR;
|
import static org.apache.solr.common.cloud.ZkStateReader.REPLICATION_FACTOR;
|
||||||
|
@ -59,371 +37,162 @@ import static org.apache.solr.common.params.ShardParams._ROUTE_;
|
||||||
/**
|
/**
|
||||||
* Tests the Custom Sharding API.
|
* Tests the Custom Sharding API.
|
||||||
*/
|
*/
|
||||||
@Slow
|
public class CustomCollectionTest extends SolrCloudTestCase {
|
||||||
@SuppressSSL(bugUrl = "https://issues.apache.org/jira/browse/SOLR-5776")
|
|
||||||
public class CustomCollectionTest extends AbstractFullDistribZkTestBase {
|
|
||||||
|
|
||||||
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
|
private static final int NODE_COUNT = 4;
|
||||||
|
|
||||||
protected String getSolrXml() {
|
@BeforeClass
|
||||||
return "solr.xml";
|
public static void setupCluster() throws Exception {
|
||||||
|
configureCluster(NODE_COUNT)
|
||||||
|
.addConfig("conf", configset("cloud-dynamic"))
|
||||||
|
.configure();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Before
|
||||||
public CustomCollectionTest() {
|
public void ensureClusterEmpty() throws Exception {
|
||||||
sliceCount = 2;
|
cluster.deleteAllCollections();
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void setDistributedParams(ModifiableSolrParams params) {
|
|
||||||
|
|
||||||
if (r.nextBoolean()) {
|
|
||||||
// don't set shards, let that be figured out from the cloud state
|
|
||||||
} else {
|
|
||||||
// use shard ids rather than physical locations
|
|
||||||
StringBuilder sb = new StringBuilder();
|
|
||||||
for (int i = 0; i < getShardCount(); i++) {
|
|
||||||
if (i > 0)
|
|
||||||
sb.append(',');
|
|
||||||
sb.append("shard" + (i + 3));
|
|
||||||
}
|
|
||||||
params.set("shards", sb.toString());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
@ShardsFixed(num = 4)
|
public void testCustomCollectionsAPI() throws Exception {
|
||||||
public void test() throws Exception {
|
|
||||||
testCustomCollectionsAPI();
|
|
||||||
testRouteFieldForHashRouter();
|
|
||||||
testCreateShardRepFactor();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
final String collection = "implicitcoll";
|
||||||
private void testCustomCollectionsAPI() throws Exception {
|
|
||||||
String COLL_PREFIX = "implicitcoll";
|
|
||||||
|
|
||||||
// TODO: fragile - because we dont pass collection.confName, it will only
|
|
||||||
// find a default if a conf set with a name matching the collection name is found, or
|
|
||||||
// if there is only one conf set. That and the fact that other tests run first in this
|
|
||||||
// env make this pretty fragile
|
|
||||||
|
|
||||||
// create new collections rapid fire
|
|
||||||
Map<String,List<Integer>> collectionInfos = new HashMap<>();
|
|
||||||
int replicationFactor = TestUtil.nextInt(random(), 0, 3) + 2;
|
int replicationFactor = TestUtil.nextInt(random(), 0, 3) + 2;
|
||||||
|
|
||||||
int cnt = random().nextInt(6) + 1;
|
|
||||||
|
|
||||||
for (int i = 0; i < cnt; i++) {
|
|
||||||
int numShards = 3;
|
int numShards = 3;
|
||||||
int maxShardsPerNode = ((((numShards+1) * replicationFactor) / getCommonCloudSolrClient()
|
int maxShardsPerNode = (((numShards + 1) * replicationFactor) / NODE_COUNT) + 1;
|
||||||
.getZkStateReader().getClusterState().getLiveNodes().size())) + 1;
|
|
||||||
|
|
||||||
|
CollectionAdminRequest.createCollectionWithImplicitRouter(collection, "conf", "a,b,c", replicationFactor)
|
||||||
|
.setMaxShardsPerNode(maxShardsPerNode)
|
||||||
|
.process(cluster.getSolrClient());
|
||||||
|
|
||||||
CloudSolrClient client = null;
|
DocCollection coll = getCollectionState(collection);
|
||||||
try {
|
|
||||||
if (i == 0) {
|
|
||||||
// Test if we can create a collection through CloudSolrServer where
|
|
||||||
// you havnt set default-collection
|
|
||||||
// This is nice because you want to be able to create you first
|
|
||||||
// collection using CloudSolrServer, and in such case there is
|
|
||||||
// nothing reasonable to set as default-collection
|
|
||||||
client = createCloudClient(null);
|
|
||||||
} else if (i == 1) {
|
|
||||||
// Test if we can create a collection through CloudSolrServer where
|
|
||||||
// you have set default-collection to a non-existing collection
|
|
||||||
// This is nice because you want to be able to create you first
|
|
||||||
// collection using CloudSolrServer, and in such case there is
|
|
||||||
// nothing reasonable to set as default-collection, but you might want
|
|
||||||
// to use the same CloudSolrServer throughout the entire
|
|
||||||
// lifetime of your client-application, so it is nice to be able to
|
|
||||||
// set a default-collection on this CloudSolrServer once and for all
|
|
||||||
// and use this CloudSolrServer to create the collection
|
|
||||||
client = createCloudClient(COLL_PREFIX + i);
|
|
||||||
}
|
|
||||||
|
|
||||||
Map<String, Object> props = Utils.makeMap(
|
|
||||||
"router.name", ImplicitDocRouter.NAME,
|
|
||||||
REPLICATION_FACTOR, replicationFactor,
|
|
||||||
MAX_SHARDS_PER_NODE, maxShardsPerNode,
|
|
||||||
SHARDS_PROP, "a,b,c");
|
|
||||||
|
|
||||||
createCollection(collectionInfos, COLL_PREFIX + i,props,client);
|
|
||||||
} finally {
|
|
||||||
if (client != null) client.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Set<Entry<String,List<Integer>>> collectionInfosEntrySet = collectionInfos.entrySet();
|
|
||||||
for (Entry<String,List<Integer>> entry : collectionInfosEntrySet) {
|
|
||||||
String collection = entry.getKey();
|
|
||||||
List<Integer> list = entry.getValue();
|
|
||||||
checkForCollection(collection, list, null);
|
|
||||||
|
|
||||||
String url = getUrlFromZk(getCommonCloudSolrClient().getZkStateReader().getClusterState(), collection);
|
|
||||||
|
|
||||||
try (HttpSolrClient collectionClient = getHttpSolrClient(url)) {
|
|
||||||
// poll for a second - it can take a moment before we are ready to serve
|
|
||||||
waitForNon403or404or503(collectionClient);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
ZkStateReader zkStateReader = getCommonCloudSolrClient().getZkStateReader();
|
|
||||||
for (int j = 0; j < cnt; j++) {
|
|
||||||
waitForRecoveriesToFinish(COLL_PREFIX + j, zkStateReader, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
ClusterState clusterState = zkStateReader.getClusterState();
|
|
||||||
|
|
||||||
DocCollection coll = clusterState.getCollection(COLL_PREFIX + 0);
|
|
||||||
assertEquals("implicit", ((Map) coll.get(DOC_ROUTER)).get("name"));
|
assertEquals("implicit", ((Map) coll.get(DOC_ROUTER)).get("name"));
|
||||||
assertNotNull(coll.getStr(REPLICATION_FACTOR));
|
assertNotNull(coll.getStr(REPLICATION_FACTOR));
|
||||||
assertNotNull(coll.getStr(MAX_SHARDS_PER_NODE));
|
assertNotNull(coll.getStr(MAX_SHARDS_PER_NODE));
|
||||||
assertNull("A shard of a Collection configured with implicit router must have null range",
|
assertNull("A shard of a Collection configured with implicit router must have null range",
|
||||||
coll.getSlice("a").getRange());
|
coll.getSlice("a").getRange());
|
||||||
|
|
||||||
List<String> collectionNameList = new ArrayList<>();
|
new UpdateRequest()
|
||||||
collectionNameList.addAll(collectionInfos.keySet());
|
.add("id", "6")
|
||||||
log.info("Collections created : "+collectionNameList );
|
.add("id", "7")
|
||||||
|
.add("id", "8")
|
||||||
|
.withRoute("a")
|
||||||
|
.commit(cluster.getSolrClient(), collection);
|
||||||
|
|
||||||
String collectionName = collectionNameList.get(random().nextInt(collectionNameList.size()));
|
assertEquals(3, cluster.getSolrClient().query(collection, new SolrQuery("*:*")).getResults().getNumFound());
|
||||||
|
assertEquals(0, cluster.getSolrClient().query(collection, new SolrQuery("*:*").setParam(_ROUTE_, "b")).getResults().getNumFound());
|
||||||
|
assertEquals(3, cluster.getSolrClient().query(collection, new SolrQuery("*:*").setParam(_ROUTE_, "a")).getResults().getNumFound());
|
||||||
|
|
||||||
String url = getUrlFromZk(getCommonCloudSolrClient().getZkStateReader().getClusterState(), collectionName);
|
cluster.getSolrClient().deleteByQuery(collection, "*:*");
|
||||||
|
cluster.getSolrClient().commit(collection, true, true);
|
||||||
|
assertEquals(0, cluster.getSolrClient().query(collection, new SolrQuery("*:*")).getResults().getNumFound());
|
||||||
|
|
||||||
String shard_fld = "shard_s";
|
new UpdateRequest()
|
||||||
try (HttpSolrClient collectionClient = getHttpSolrClient(url)) {
|
.add("id", "9")
|
||||||
|
.add("id", "10")
|
||||||
|
.add("id", "11")
|
||||||
|
.withRoute("c")
|
||||||
|
.commit(cluster.getSolrClient(), collection);
|
||||||
|
|
||||||
// lets try and use the solrj client to index a couple documents
|
assertEquals(3, cluster.getSolrClient().query(collection, new SolrQuery("*:*")).getResults().getNumFound());
|
||||||
|
assertEquals(0, cluster.getSolrClient().query(collection, new SolrQuery("*:*").setParam(_ROUTE_, "a")).getResults().getNumFound());
|
||||||
collectionClient.add(getDoc(id, 6, i1, -600, tlong, 600, t1,
|
assertEquals(3, cluster.getSolrClient().query(collection, new SolrQuery("*:*").setParam(_ROUTE_, "c")).getResults().getNumFound());
|
||||||
"humpty dumpy sat on a wall", _ROUTE_,"a"));
|
|
||||||
|
|
||||||
collectionClient.add(getDoc(id, 7, i1, -600, tlong, 600, t1,
|
|
||||||
"humpty dumpy3 sat on a walls", _ROUTE_,"a"));
|
|
||||||
|
|
||||||
collectionClient.add(getDoc(id, 8, i1, -600, tlong, 600, t1,
|
|
||||||
"humpty dumpy2 sat on a walled", _ROUTE_,"a"));
|
|
||||||
|
|
||||||
collectionClient.commit();
|
|
||||||
|
|
||||||
assertEquals(3, collectionClient.query(new SolrQuery("*:*")).getResults().getNumFound());
|
|
||||||
assertEquals(0, collectionClient.query(new SolrQuery("*:*").setParam(_ROUTE_,"b")).getResults().getNumFound());
|
|
||||||
assertEquals(3, collectionClient.query(new SolrQuery("*:*").setParam(_ROUTE_,"a")).getResults().getNumFound());
|
|
||||||
|
|
||||||
collectionClient.deleteByQuery("*:*");
|
|
||||||
collectionClient.commit(true,true);
|
|
||||||
assertEquals(0, collectionClient.query(new SolrQuery("*:*")).getResults().getNumFound());
|
|
||||||
|
|
||||||
UpdateRequest up = new UpdateRequest();
|
|
||||||
up.setParam(_ROUTE_, "c");
|
|
||||||
up.setParam("commit","true");
|
|
||||||
|
|
||||||
up.add(getDoc(id, 9, i1, -600, tlong, 600, t1,
|
|
||||||
"humpty dumpy sat on a wall"));
|
|
||||||
up.add(getDoc(id, 10, i1, -600, tlong, 600, t1,
|
|
||||||
"humpty dumpy3 sat on a walls"));
|
|
||||||
up.add(getDoc(id, 11, i1, -600, tlong, 600, t1,
|
|
||||||
"humpty dumpy2 sat on a walled"));
|
|
||||||
|
|
||||||
collectionClient.request(up);
|
|
||||||
|
|
||||||
assertEquals(3, collectionClient.query(new SolrQuery("*:*")).getResults().getNumFound());
|
|
||||||
assertEquals(0, collectionClient.query(new SolrQuery("*:*").setParam(_ROUTE_,"a")).getResults().getNumFound());
|
|
||||||
assertEquals(3, collectionClient.query(new SolrQuery("*:*").setParam(_ROUTE_,"c")).getResults().getNumFound());
|
|
||||||
|
|
||||||
//Testing CREATESHARD
|
//Testing CREATESHARD
|
||||||
ModifiableSolrParams params = new ModifiableSolrParams();
|
CollectionAdminRequest.createShard(collection, "x")
|
||||||
params.set("action", CollectionAction.CREATESHARD.toString());
|
.process(cluster.getSolrClient());
|
||||||
params.set("collection", collectionName);
|
waitForState("Expected shard 'x' to be active", collection, (n, c) -> {
|
||||||
params.set("shard", "x");
|
if (c.getSlice("x") == null)
|
||||||
SolrRequest request = new QueryRequest(params);
|
return false;
|
||||||
request.setPath("/admin/collections");
|
for (Replica r : c.getSlice("x")) {
|
||||||
try (SolrClient server = createNewSolrClient("", getBaseUrl((HttpSolrClient) clients.get(0)))) {
|
if (r.getState() != Replica.State.ACTIVE)
|
||||||
server.request(request);
|
return false;
|
||||||
}
|
}
|
||||||
waitForCollection(zkStateReader,collectionName,4);
|
return true;
|
||||||
//wait for all the replicas to become active
|
});
|
||||||
int attempts = 0;
|
|
||||||
while(true){
|
|
||||||
if(attempts>30 ) fail("Not enough active replicas in the shard 'x'");
|
|
||||||
attempts++;
|
|
||||||
int activeReplicaCount = 0;
|
|
||||||
for (Replica x : zkStateReader.getClusterState().getCollection(collectionName).getSlice("x").getReplicas()) {
|
|
||||||
if (x.getState() == Replica.State.ACTIVE) {
|
|
||||||
activeReplicaCount++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Thread.sleep(500);
|
|
||||||
if(activeReplicaCount >= replicationFactor) break;
|
|
||||||
}
|
|
||||||
log.info(zkStateReader.getClusterState().toString());
|
|
||||||
|
|
||||||
collectionClient.add(getDoc(id, 66, i1, -600, tlong, 600, t1,
|
new UpdateRequest()
|
||||||
"humpty dumpy sat on a wall", _ROUTE_,"x"));
|
.add("id", "66", _ROUTE_, "x")
|
||||||
collectionClient.commit();
|
.commit(cluster.getSolrClient(), collection);
|
||||||
assertEquals(1, collectionClient.query(new SolrQuery("*:*").setParam(_ROUTE_,"x")).getResults().getNumFound());
|
// TODO - the local state is cached and causes the request to fail with 'unknown shard'
|
||||||
|
// assertEquals(1, cluster.getSolrClient().query(collection, new SolrQuery("*:*").setParam(_ROUTE_, "x")).getResults().getNumFound());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testRouteFieldForImplicitRouter() throws Exception {
|
||||||
|
|
||||||
int numShards = 4;
|
int numShards = 4;
|
||||||
replicationFactor = TestUtil.nextInt(random(), 0, 3) + 2;
|
int replicationFactor = TestUtil.nextInt(random(), 0, 3) + 2;
|
||||||
int maxShardsPerNode = (((numShards * replicationFactor) / getCommonCloudSolrClient()
|
int maxShardsPerNode = ((numShards * replicationFactor) / NODE_COUNT) + 1;
|
||||||
.getZkStateReader().getClusterState().getLiveNodes().size())) + 1;
|
String shard_fld = "shard_s";
|
||||||
|
|
||||||
try (CloudSolrClient client = createCloudClient(null)) {
|
final String collection = "withShardField";
|
||||||
Map<String, Object> props = Utils.makeMap(
|
|
||||||
"router.name", ImplicitDocRouter.NAME,
|
CollectionAdminRequest.createCollectionWithImplicitRouter(collection, "conf", "a,b,c,d", replicationFactor)
|
||||||
REPLICATION_FACTOR, replicationFactor,
|
.setMaxShardsPerNode(maxShardsPerNode)
|
||||||
MAX_SHARDS_PER_NODE, maxShardsPerNode,
|
.setRouterField(shard_fld)
|
||||||
SHARDS_PROP, "a,b,c,d",
|
.process(cluster.getSolrClient());
|
||||||
"router.field", shard_fld);
|
|
||||||
|
new UpdateRequest()
|
||||||
|
.add("id", "6", shard_fld, "a")
|
||||||
|
.add("id", "7", shard_fld, "a")
|
||||||
|
.add("id", "8", shard_fld, "b")
|
||||||
|
.commit(cluster.getSolrClient(), collection);
|
||||||
|
|
||||||
|
assertEquals(3, cluster.getSolrClient().query(collection, new SolrQuery("*:*")).getResults().getNumFound());
|
||||||
|
assertEquals(1, cluster.getSolrClient().query(collection, new SolrQuery("*:*").setParam(_ROUTE_, "b")).getResults().getNumFound());
|
||||||
|
assertEquals(2, cluster.getSolrClient().query(collection, new SolrQuery("*:*").setParam(_ROUTE_, "a")).getResults().getNumFound());
|
||||||
|
|
||||||
collectionName = COLL_PREFIX + "withShardField";
|
|
||||||
createCollection(collectionInfos, collectionName,props,client);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
List<Integer> list = collectionInfos.get(collectionName);
|
@Test
|
||||||
checkForCollection(collectionName, list, null);
|
public void testRouteFieldForHashRouter()throws Exception{
|
||||||
|
|
||||||
|
|
||||||
url = getUrlFromZk(getCommonCloudSolrClient().getZkStateReader().getClusterState(), collectionName);
|
|
||||||
}
|
|
||||||
|
|
||||||
try (HttpSolrClient collectionClient = getHttpSolrClient(url)) {
|
|
||||||
// poll for a second - it can take a moment before we are ready to serve
|
|
||||||
waitForNon403or404or503(collectionClient);
|
|
||||||
}
|
|
||||||
|
|
||||||
try (HttpSolrClient collectionClient = getHttpSolrClient(url)) {
|
|
||||||
// lets try and use the solrj client to index a couple documents
|
|
||||||
|
|
||||||
collectionClient.add(getDoc(id, 6, i1, -600, tlong, 600, t1,
|
|
||||||
"humpty dumpy sat on a wall", shard_fld,"a"));
|
|
||||||
|
|
||||||
collectionClient.add(getDoc(id, 7, i1, -600, tlong, 600, t1,
|
|
||||||
"humpty dumpy3 sat on a walls", shard_fld,"a"));
|
|
||||||
|
|
||||||
collectionClient.add(getDoc(id, 8, i1, -600, tlong, 600, t1,
|
|
||||||
"humpty dumpy2 sat on a walled", shard_fld,"a"));
|
|
||||||
|
|
||||||
collectionClient.commit();
|
|
||||||
|
|
||||||
assertEquals(3, collectionClient.query(new SolrQuery("*:*")).getResults().getNumFound());
|
|
||||||
assertEquals(0, collectionClient.query(new SolrQuery("*:*").setParam(_ROUTE_,"b")).getResults().getNumFound());
|
|
||||||
//TODO debug the following case
|
|
||||||
assertEquals(3, collectionClient.query(new SolrQuery("*:*").setParam(_ROUTE_, "a")).getResults().getNumFound());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void testRouteFieldForHashRouter()throws Exception{
|
|
||||||
String collectionName = "routeFieldColl";
|
String collectionName = "routeFieldColl";
|
||||||
int numShards = 4;
|
int numShards = 4;
|
||||||
int replicationFactor = 2;
|
int replicationFactor = 2;
|
||||||
int maxShardsPerNode = (((numShards * replicationFactor) / getCommonCloudSolrClient()
|
int maxShardsPerNode = ((numShards * replicationFactor) / NODE_COUNT) + 1;
|
||||||
.getZkStateReader().getClusterState().getLiveNodes().size())) + 1;
|
|
||||||
|
|
||||||
HashMap<String, List<Integer>> collectionInfos = new HashMap<>();
|
|
||||||
String shard_fld = "shard_s";
|
String shard_fld = "shard_s";
|
||||||
try (CloudSolrClient client = createCloudClient(null)) {
|
|
||||||
Map<String, Object> props = Utils.makeMap(
|
|
||||||
REPLICATION_FACTOR, replicationFactor,
|
|
||||||
MAX_SHARDS_PER_NODE, maxShardsPerNode,
|
|
||||||
NUM_SLICES, numShards,
|
|
||||||
"router.field", shard_fld);
|
|
||||||
|
|
||||||
createCollection(collectionInfos, collectionName,props,client);
|
CollectionAdminRequest.createCollection(collectionName, "conf", numShards, replicationFactor)
|
||||||
|
.setMaxShardsPerNode(maxShardsPerNode)
|
||||||
|
.setRouterField(shard_fld)
|
||||||
|
.process(cluster.getSolrClient());
|
||||||
|
|
||||||
|
new UpdateRequest()
|
||||||
|
.add("id", "6", shard_fld, "a")
|
||||||
|
.add("id", "7", shard_fld, "a")
|
||||||
|
.add("id", "8", shard_fld, "b")
|
||||||
|
.commit(cluster.getSolrClient(), collectionName);
|
||||||
|
|
||||||
|
assertEquals(3, cluster.getSolrClient().query(collectionName, new SolrQuery("*:*")).getResults().getNumFound());
|
||||||
|
assertEquals(2, cluster.getSolrClient().query(collectionName, new SolrQuery("*:*").setParam(_ROUTE_, "a")).getResults().getNumFound());
|
||||||
|
assertEquals(1, cluster.getSolrClient().query(collectionName, new SolrQuery("*:*").setParam(_ROUTE_, "b")).getResults().getNumFound());
|
||||||
|
assertEquals(0, cluster.getSolrClient().query(collectionName, new SolrQuery("*:*").setParam(_ROUTE_, "c")).getResults().getNumFound());
|
||||||
|
|
||||||
|
|
||||||
|
cluster.getSolrClient().deleteByQuery(collectionName, "*:*");
|
||||||
|
cluster.getSolrClient().commit(collectionName);
|
||||||
|
|
||||||
|
cluster.getSolrClient().add(collectionName, new SolrInputDocument("id", "100", shard_fld, "c!doc1"));
|
||||||
|
cluster.getSolrClient().commit(collectionName);
|
||||||
|
assertEquals(1, cluster.getSolrClient().query(collectionName, new SolrQuery("*:*").setParam(_ROUTE_, "c!")).getResults().getNumFound());
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
List<Integer> list = collectionInfos.get(collectionName);
|
@Test
|
||||||
checkForCollection(collectionName, list, null);
|
public void testCreateShardRepFactor() throws Exception {
|
||||||
|
final String collectionName = "testCreateShardRepFactor";
|
||||||
|
CollectionAdminRequest.createCollectionWithImplicitRouter(collectionName, "conf", "a,b", 1)
|
||||||
|
.process(cluster.getSolrClient());
|
||||||
|
|
||||||
|
CollectionAdminRequest.createShard(collectionName, "x")
|
||||||
|
.process(cluster.getSolrClient());
|
||||||
|
|
||||||
String url = getUrlFromZk(getCommonCloudSolrClient().getZkStateReader().getClusterState(), collectionName);
|
waitForState("Not enough active replicas in shard 'x'", collectionName, (n, c) -> {
|
||||||
|
return c.getSlice("x").getReplicas().size() == 1;
|
||||||
|
});
|
||||||
|
|
||||||
try (HttpSolrClient collectionClient = getHttpSolrClient(url)) {
|
|
||||||
// poll for a second - it can take a moment before we are ready to serve
|
|
||||||
waitForNon403or404or503(collectionClient);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
try (HttpSolrClient collectionClient = getHttpSolrClient(url)) {
|
|
||||||
// lets try and use the solrj client to index a couple documents
|
|
||||||
|
|
||||||
collectionClient.add(getDoc(id, 6, i1, -600, tlong, 600, t1,
|
|
||||||
"humpty dumpy sat on a wall", shard_fld,"a"));
|
|
||||||
|
|
||||||
collectionClient.add(getDoc(id, 7, i1, -600, tlong, 600, t1,
|
|
||||||
"humpty dumpy3 sat on a walls", shard_fld,"a"));
|
|
||||||
|
|
||||||
collectionClient.add(getDoc(id, 8, i1, -600, tlong, 600, t1,
|
|
||||||
"humpty dumpy2 sat on a walled", shard_fld,"a"));
|
|
||||||
|
|
||||||
collectionClient.commit();
|
|
||||||
|
|
||||||
assertEquals(3, collectionClient.query(new SolrQuery("*:*")).getResults().getNumFound());
|
|
||||||
//TODO debug the following case
|
|
||||||
assertEquals(3, collectionClient.query(new SolrQuery("*:*").setParam(_ROUTE_, "a")).getResults().getNumFound());
|
|
||||||
|
|
||||||
collectionClient.deleteByQuery("*:*");
|
|
||||||
collectionClient.commit();
|
|
||||||
|
|
||||||
collectionClient.add (getDoc( id,100,shard_fld, "b!doc1"));
|
|
||||||
collectionClient.commit();
|
|
||||||
assertEquals(1, collectionClient.query(new SolrQuery("*:*").setParam(_ROUTE_, "b!")).getResults().getNumFound());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void testCreateShardRepFactor() throws Exception {
|
|
||||||
String collectionName = "testCreateShardRepFactor";
|
|
||||||
HashMap<String, List<Integer>> collectionInfos = new HashMap<>();
|
|
||||||
try (CloudSolrClient client = createCloudClient(null)) {
|
|
||||||
Map<String, Object> props = Utils.makeMap(
|
|
||||||
REPLICATION_FACTOR, 1,
|
|
||||||
MAX_SHARDS_PER_NODE, 5,
|
|
||||||
NUM_SLICES, 2,
|
|
||||||
"shards", "a,b",
|
|
||||||
"router.name", "implicit");
|
|
||||||
|
|
||||||
createCollection(collectionInfos, collectionName, props, client);
|
|
||||||
}
|
|
||||||
ZkStateReader zkStateReader = getCommonCloudSolrClient().getZkStateReader();
|
|
||||||
waitForRecoveriesToFinish(collectionName, zkStateReader, false);
|
|
||||||
|
|
||||||
ModifiableSolrParams params = new ModifiableSolrParams();
|
|
||||||
params.set("action", CollectionAction.CREATESHARD.toString());
|
|
||||||
params.set("collection", collectionName);
|
|
||||||
params.set("shard", "x");
|
|
||||||
SolrRequest request = new QueryRequest(params);
|
|
||||||
request.setPath("/admin/collections");
|
|
||||||
|
|
||||||
try (SolrClient server = createNewSolrClient("", getBaseUrl((HttpSolrClient) clients.get(0)))) {
|
|
||||||
server.request(request);
|
|
||||||
}
|
|
||||||
|
|
||||||
waitForRecoveriesToFinish(collectionName, zkStateReader, false);
|
|
||||||
|
|
||||||
int replicaCount = 0;
|
|
||||||
int attempts = 0;
|
|
||||||
while (true) {
|
|
||||||
if (attempts > 30) fail("Not enough active replicas in the shard 'x'");
|
|
||||||
attempts++;
|
|
||||||
replicaCount = zkStateReader.getClusterState().getSlice(collectionName, "x").getReplicas().size();
|
|
||||||
if (replicaCount >= 1) break;
|
|
||||||
Thread.sleep(500);
|
|
||||||
}
|
|
||||||
|
|
||||||
assertEquals("CREATESHARD API created more than replicationFactor number of replicas", 1, replicaCount);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected QueryResponse queryServer(ModifiableSolrParams params) throws SolrServerException, IOException {
|
|
||||||
|
|
||||||
if (r.nextBoolean())
|
|
||||||
return super.queryServer(params);
|
|
||||||
|
|
||||||
if (r.nextBoolean())
|
|
||||||
params.set("collection",DEFAULT_COLLECTION);
|
|
||||||
|
|
||||||
QueryResponse rsp = getCommonCloudSolrClient().query(params);
|
|
||||||
return rsp;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -51,6 +51,7 @@ public class MigrateRouteKeyTest extends SolrCloudTestCase {
|
||||||
|
|
||||||
if (usually()) {
|
if (usually()) {
|
||||||
CollectionAdminRequest.setClusterProperty("legacyCloud", "false").process(cluster.getSolrClient());
|
CollectionAdminRequest.setClusterProperty("legacyCloud", "false").process(cluster.getSolrClient());
|
||||||
|
log.info("Using legacyCloud=false for cluster");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -16,45 +16,58 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.cloud;
|
package org.apache.solr.cloud;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.lang.invoke.MethodHandles;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
import org.apache.lucene.util.LuceneTestCase.Slow;
|
import org.apache.lucene.util.LuceneTestCase.Slow;
|
||||||
import org.apache.solr.client.solrj.SolrQuery;
|
import org.apache.solr.client.solrj.SolrQuery;
|
||||||
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.common.SolrInputDocument;
|
import org.apache.solr.client.solrj.impl.HttpSolrClient;
|
||||||
import org.apache.solr.common.cloud.ZkStateReader;
|
import org.apache.solr.client.solrj.request.CollectionAdminRequest;
|
||||||
|
import org.apache.solr.client.solrj.request.UpdateRequest;
|
||||||
|
import org.apache.solr.common.cloud.DocCollection;
|
||||||
|
import org.apache.solr.common.cloud.Replica;
|
||||||
|
import org.apache.solr.common.cloud.Slice;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
@Slow
|
@Slow
|
||||||
public class RecoveryZkTest extends AbstractFullDistribZkTestBase {
|
public class RecoveryZkTest extends SolrCloudTestCase {
|
||||||
|
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
|
||||||
|
|
||||||
|
@BeforeClass
|
||||||
|
public static void setupCluster() throws Exception {
|
||||||
|
configureCluster(2)
|
||||||
|
.addConfig("conf", configset("cloud-minimal"))
|
||||||
|
.configure();
|
||||||
|
}
|
||||||
|
|
||||||
//private static final String DISTRIB_UPDATE_CHAIN = "distrib-update-chain";
|
|
||||||
private StoppableIndexingThread indexThread;
|
private StoppableIndexingThread indexThread;
|
||||||
private StoppableIndexingThread indexThread2;
|
private StoppableIndexingThread indexThread2;
|
||||||
|
|
||||||
public RecoveryZkTest() {
|
@After
|
||||||
super();
|
public void stopThreads() throws InterruptedException {
|
||||||
sliceCount = 1;
|
indexThread.safeStop();
|
||||||
fixShardCount(2);
|
indexThread2.safeStop();
|
||||||
schemaString = "schema15.xml"; // we need a string id
|
indexThread.join();
|
||||||
}
|
indexThread2.join();
|
||||||
|
|
||||||
public static String[] fieldNames = new String[]{"f_i", "f_f", "f_d", "f_l", "f_dt"};
|
|
||||||
public static RandVal[] randVals = new RandVal[]{rint, rfloat, rdouble, rlong, rdate};
|
|
||||||
|
|
||||||
protected String[] getFieldNames() {
|
|
||||||
return fieldNames;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected RandVal[] getRandValues() {
|
|
||||||
return randVals;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void test() throws Exception {
|
public void test() throws Exception {
|
||||||
handle.clear();
|
|
||||||
handle.put("timestamp", SKIPVAL);
|
final String collection = "recoverytest";
|
||||||
|
|
||||||
|
CollectionAdminRequest.createCollection(collection, "conf", 1, 2)
|
||||||
|
.setMaxShardsPerNode(1)
|
||||||
|
.process(cluster.getSolrClient());
|
||||||
|
waitForState("Expected a collection with one shard and two replicas", collection, clusterShape(1, 2));
|
||||||
|
cluster.getSolrClient().setDefaultCollection(collection);
|
||||||
|
|
||||||
// start a couple indexing threads
|
// start a couple indexing threads
|
||||||
|
|
||||||
|
@ -67,12 +80,12 @@ public class RecoveryZkTest extends AbstractFullDistribZkTestBase {
|
||||||
} else {
|
} else {
|
||||||
maxDoc = maxDocNightlyList[random().nextInt(maxDocList.length - 1)];
|
maxDoc = maxDocNightlyList[random().nextInt(maxDocList.length - 1)];
|
||||||
}
|
}
|
||||||
|
log.info("Indexing {} documents", maxDoc);
|
||||||
|
|
||||||
indexThread = new StoppableIndexingThread(controlClient, cloudClient, "1", true, maxDoc, 1, true);
|
indexThread = new StoppableIndexingThread(null, cluster.getSolrClient(), "1", true, maxDoc, 1, true);
|
||||||
indexThread.start();
|
indexThread.start();
|
||||||
|
|
||||||
indexThread2 = new StoppableIndexingThread(controlClient, cloudClient, "2", true, maxDoc, 1, true);
|
indexThread2 = new StoppableIndexingThread(null, cluster.getSolrClient(), "2", true, maxDoc, 1, true);
|
||||||
|
|
||||||
indexThread2.start();
|
indexThread2.start();
|
||||||
|
|
||||||
// give some time to index...
|
// give some time to index...
|
||||||
|
@ -80,18 +93,21 @@ public class RecoveryZkTest extends AbstractFullDistribZkTestBase {
|
||||||
Thread.sleep(waitTimes[random().nextInt(waitTimes.length - 1)]);
|
Thread.sleep(waitTimes[random().nextInt(waitTimes.length - 1)]);
|
||||||
|
|
||||||
// bring shard replica down
|
// bring shard replica down
|
||||||
JettySolrRunner replica = chaosMonkey.stopShard("shard1", 1).jetty;
|
DocCollection state = getCollectionState(collection);
|
||||||
|
Replica leader = state.getLeader("shard1");
|
||||||
|
Replica replica = getRandomReplica(state.getSlice("shard1"), (r) -> leader != r);
|
||||||
|
|
||||||
|
JettySolrRunner jetty = cluster.getReplicaJetty(replica);
|
||||||
|
jetty.stop();
|
||||||
|
|
||||||
// wait a moment - lets allow some docs to be indexed so replication time is non 0
|
// wait a moment - lets allow some docs to be indexed so replication time is non 0
|
||||||
Thread.sleep(waitTimes[random().nextInt(waitTimes.length - 1)]);
|
Thread.sleep(waitTimes[random().nextInt(waitTimes.length - 1)]);
|
||||||
|
|
||||||
// bring shard replica up
|
// bring shard replica up
|
||||||
replica.start();
|
jetty.start();
|
||||||
|
|
||||||
// make sure replication can start
|
// make sure replication can start
|
||||||
Thread.sleep(3000);
|
Thread.sleep(3000);
|
||||||
ZkStateReader zkStateReader = cloudClient.getZkStateReader();
|
|
||||||
|
|
||||||
// stop indexing threads
|
// stop indexing threads
|
||||||
indexThread.safeStop();
|
indexThread.safeStop();
|
||||||
|
@ -100,68 +116,34 @@ public class RecoveryZkTest extends AbstractFullDistribZkTestBase {
|
||||||
indexThread.join();
|
indexThread.join();
|
||||||
indexThread2.join();
|
indexThread2.join();
|
||||||
|
|
||||||
Thread.sleep(1000);
|
new UpdateRequest()
|
||||||
|
.commit(cluster.getSolrClient(), collection);
|
||||||
|
|
||||||
waitForThingsToLevelOut(120);
|
cluster.getSolrClient().waitForState(collection, 120, TimeUnit.SECONDS, clusterShape(1, 2));
|
||||||
|
|
||||||
Thread.sleep(2000);
|
|
||||||
|
|
||||||
waitForThingsToLevelOut(30);
|
|
||||||
|
|
||||||
Thread.sleep(5000);
|
|
||||||
|
|
||||||
waitForRecoveriesToFinish(DEFAULT_COLLECTION, zkStateReader, false, true);
|
|
||||||
|
|
||||||
// test that leader and replica have same doc count
|
// test that leader and replica have same doc count
|
||||||
|
state = getCollectionState(collection);
|
||||||
|
assertShardConsistency(state.getSlice("shard1"), true);
|
||||||
|
|
||||||
String fail = checkShardConsistency("shard1", false, false);
|
|
||||||
if (fail != null) {
|
|
||||||
fail(fail);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
SolrQuery query = new SolrQuery("*:*");
|
private void assertShardConsistency(Slice shard, boolean expectDocs) throws Exception {
|
||||||
query.setParam("distrib", "false");
|
List<Replica> replicas = shard.getReplicas(r -> r.getState() == Replica.State.ACTIVE);
|
||||||
long client1Docs = shardToJetty.get("shard1").get(0).client.solrClient.query(query).getResults().getNumFound();
|
long[] numCounts = new long[replicas.size()];
|
||||||
long client2Docs = shardToJetty.get("shard1").get(1).client.solrClient.query(query).getResults().getNumFound();
|
int i = 0;
|
||||||
|
for (Replica replica : replicas) {
|
||||||
assertTrue(client1Docs > 0);
|
try (HttpSolrClient client = new HttpSolrClient.Builder(replica.getCoreUrl())
|
||||||
assertEquals(client1Docs, client2Docs);
|
.withHttpClient(cluster.getSolrClient().getHttpClient()).build()) {
|
||||||
|
numCounts[i] = client.query(new SolrQuery("*:*").add("distrib", "false")).getResults().getNumFound();
|
||||||
// won't always pass yet...
|
i++;
|
||||||
//query("q", "*:*", "sort", "id desc");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected void indexDoc(SolrInputDocument doc) throws IOException,
|
|
||||||
SolrServerException {
|
|
||||||
controlClient.add(doc);
|
|
||||||
|
|
||||||
// UpdateRequest ureq = new UpdateRequest();
|
|
||||||
// ureq.add(doc);
|
|
||||||
// ureq.setParam("update.chain", DISTRIB_UPDATE_CHAIN);
|
|
||||||
// ureq.process(cloudClient);
|
|
||||||
cloudClient.add(doc);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void distribTearDown() throws Exception {
|
|
||||||
// make sure threads have been stopped...
|
|
||||||
indexThread.safeStop();
|
|
||||||
indexThread2.safeStop();
|
|
||||||
|
|
||||||
indexThread.join();
|
|
||||||
indexThread2.join();
|
|
||||||
|
|
||||||
super.distribTearDown();
|
|
||||||
}
|
|
||||||
|
|
||||||
// skip the randoms - they can deadlock...
|
|
||||||
@Override
|
|
||||||
protected void indexr(Object... fields) throws Exception {
|
|
||||||
SolrInputDocument doc = new SolrInputDocument();
|
|
||||||
addFields(doc, fields);
|
|
||||||
addFields(doc, "rnd_b", true);
|
|
||||||
indexDoc(doc);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
for (int j = 1; j < replicas.size(); j++) {
|
||||||
|
if (numCounts[j] != numCounts[j - 1])
|
||||||
|
fail("Mismatch in counts between replicas"); // TODO improve this!
|
||||||
|
if (numCounts[j] == 0 && expectDocs)
|
||||||
|
fail("Expected docs on shard " + shard.getName() + " but found none");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
|
@ -93,7 +93,8 @@ public class ShardSplitTest extends BasicDistributedZkTest {
|
||||||
|
|
||||||
if (usually()) {
|
if (usually()) {
|
||||||
log.info("Using legacyCloud=false for cluster");
|
log.info("Using legacyCloud=false for cluster");
|
||||||
CollectionsAPIDistributedZkTest.setClusterProp(cloudClient, "legacyCloud", "false");
|
CollectionAdminRequest.setClusterProperty(ZkStateReader.LEGACY_CLOUD, "false")
|
||||||
|
.process(cloudClient);
|
||||||
}
|
}
|
||||||
incompleteOrOverlappingCustomRangeTest();
|
incompleteOrOverlappingCustomRangeTest();
|
||||||
splitByUniqueKeyTest();
|
splitByUniqueKeyTest();
|
||||||
|
@ -516,7 +517,8 @@ public class ShardSplitTest extends BasicDistributedZkTest {
|
||||||
|
|
||||||
if (usually()) {
|
if (usually()) {
|
||||||
log.info("Using legacyCloud=false for cluster");
|
log.info("Using legacyCloud=false for cluster");
|
||||||
CollectionsAPIDistributedZkTest.setClusterProp(cloudClient, "legacyCloud", "false");
|
CollectionAdminRequest.setClusterProperty(ZkStateReader.LEGACY_CLOUD, "false")
|
||||||
|
.process(cloudClient);
|
||||||
}
|
}
|
||||||
|
|
||||||
log.info("Starting testSplitShardWithRule");
|
log.info("Starting testSplitShardWithRule");
|
||||||
|
|
|
@ -0,0 +1,45 @@
|
||||||
|
/*
|
||||||
|
* 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.cloud;
|
||||||
|
|
||||||
|
import org.apache.solr.client.solrj.request.CollectionAdminRequest;
|
||||||
|
import org.apache.solr.common.cloud.ClusterProperties;
|
||||||
|
import org.apache.solr.common.cloud.ZkStateReader;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
public class TestClusterProperties extends SolrCloudTestCase {
|
||||||
|
|
||||||
|
@BeforeClass
|
||||||
|
public static void setupCluster() throws Exception {
|
||||||
|
configureCluster(1).configure();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testClusterProperties() throws Exception {
|
||||||
|
ClusterProperties props = new ClusterProperties(zkClient());
|
||||||
|
assertEquals("false", props.getClusterProperty(ZkStateReader.LEGACY_CLOUD, "false"));
|
||||||
|
|
||||||
|
CollectionAdminRequest.setClusterProperty(ZkStateReader.LEGACY_CLOUD, "true").process(cluster.getSolrClient());
|
||||||
|
assertEquals("true", props.getClusterProperty(ZkStateReader.LEGACY_CLOUD, "false"));
|
||||||
|
|
||||||
|
CollectionAdminRequest.setClusterProperty(ZkStateReader.LEGACY_CLOUD, "false").process(cluster.getSolrClient());
|
||||||
|
assertEquals("false", props.getClusterProperty(ZkStateReader.LEGACY_CLOUD, "true"));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,65 @@
|
||||||
|
/*
|
||||||
|
* 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.cloud;
|
||||||
|
|
||||||
|
import org.apache.solr.client.solrj.request.CollectionAdminRequest;
|
||||||
|
import org.apache.solr.common.cloud.Slice;
|
||||||
|
import org.junit.BeforeClass;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
public class TestDeleteCollectionOnDownNodes extends SolrCloudTestCase {
|
||||||
|
|
||||||
|
@BeforeClass
|
||||||
|
public static void setupCluster() throws Exception {
|
||||||
|
configureCluster(4)
|
||||||
|
.addConfig("conf", configset("cloud-minimal"))
|
||||||
|
.addConfig("conf2", configset("cloud-minimal"))
|
||||||
|
.configure();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void deleteCollectionWithDownNodes() throws Exception {
|
||||||
|
|
||||||
|
CollectionAdminRequest.createCollection("halfdeletedcollection2", "conf", 4, 3)
|
||||||
|
.setMaxShardsPerNode(3)
|
||||||
|
.process(cluster.getSolrClient());
|
||||||
|
|
||||||
|
// stop a couple nodes
|
||||||
|
cluster.stopJettySolrRunner(cluster.getRandomJetty(random()));
|
||||||
|
cluster.stopJettySolrRunner(cluster.getRandomJetty(random()));
|
||||||
|
|
||||||
|
// wait for leaders to settle out
|
||||||
|
waitForState("Timed out waiting for leader elections", "halfdeletedcollection2", (n, c) -> {
|
||||||
|
for (Slice slice : c) {
|
||||||
|
if (slice.getLeader() == null)
|
||||||
|
return false;
|
||||||
|
if (slice.getLeader().isActive(n) == false)
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
|
||||||
|
// delete the collection
|
||||||
|
CollectionAdminRequest.deleteCollection("halfdeletedcollection2").process(cluster.getSolrClient());
|
||||||
|
waitForState("Timed out waiting for collection to be deleted", "halfdeletedcollection2", (n, c) -> c == null);
|
||||||
|
|
||||||
|
assertFalse("Still found collection that should be gone",
|
||||||
|
cluster.getSolrClient().getZkStateReader().getClusterState().hasCollection("halfdeletedcollection2"));
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -16,48 +16,41 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.cloud.hdfs;
|
package org.apache.solr.cloud.hdfs;
|
||||||
|
|
||||||
import java.io.IOException;
|
import com.carrotsearch.randomizedtesting.annotations.Nightly;
|
||||||
|
import com.carrotsearch.randomizedtesting.annotations.ThreadLeakFilters;
|
||||||
import org.apache.hadoop.hdfs.MiniDFSCluster;
|
import org.apache.hadoop.hdfs.MiniDFSCluster;
|
||||||
import org.apache.lucene.util.LuceneTestCase.Slow;
|
import org.apache.lucene.util.LuceneTestCase.Slow;
|
||||||
import org.apache.solr.cloud.CollectionsAPIDistributedZkTest;
|
import org.apache.solr.cloud.CollectionsAPIDistributedZkTest;
|
||||||
import org.apache.solr.update.HdfsUpdateLog;
|
import org.apache.solr.common.cloud.ZkConfigManager;
|
||||||
import org.apache.solr.util.BadHdfsThreadsFilter;
|
import org.apache.solr.util.BadHdfsThreadsFilter;
|
||||||
import org.junit.AfterClass;
|
import org.junit.AfterClass;
|
||||||
import org.junit.BeforeClass;
|
import org.junit.BeforeClass;
|
||||||
|
|
||||||
import com.carrotsearch.randomizedtesting.annotations.Nightly;
|
|
||||||
import com.carrotsearch.randomizedtesting.annotations.ThreadLeakFilters;
|
|
||||||
|
|
||||||
@Slow
|
@Slow
|
||||||
@Nightly
|
@Nightly
|
||||||
@ThreadLeakFilters(defaultFilters = true, filters = {
|
@ThreadLeakFilters(defaultFilters = true, filters = {
|
||||||
BadHdfsThreadsFilter.class // hdfs currently leaks thread(s)
|
BadHdfsThreadsFilter.class // hdfs currently leaks thread(s)
|
||||||
})
|
})
|
||||||
public class HdfsCollectionsAPIDistributedZkTest extends CollectionsAPIDistributedZkTest {
|
public class HdfsCollectionsAPIDistributedZkTest extends CollectionsAPIDistributedZkTest {
|
||||||
|
|
||||||
private static MiniDFSCluster dfsCluster;
|
private static MiniDFSCluster dfsCluster;
|
||||||
private static long initialFailLogsCount;
|
|
||||||
|
|
||||||
@BeforeClass
|
@BeforeClass
|
||||||
public static void setupClass() throws Exception {
|
public static void setupClass() throws Exception {
|
||||||
dfsCluster = HdfsTestUtil.setupClass(createTempDir().toFile().getAbsolutePath());
|
dfsCluster = HdfsTestUtil.setupClass(createTempDir().toFile().getAbsolutePath());
|
||||||
System.setProperty("solr.hdfs.blockcache.enabled", "false");
|
System.setProperty("solr.hdfs.blockcache.blocksperbank", "2048");
|
||||||
initialFailLogsCount = HdfsUpdateLog.INIT_FAILED_LOGS_COUNT.get();
|
|
||||||
|
ZkConfigManager configManager = new ZkConfigManager(zkClient());
|
||||||
|
configManager.uploadConfigDir(configset("cloud-hdfs"), "conf");
|
||||||
|
|
||||||
|
System.setProperty("solr.hdfs.home", HdfsTestUtil.getDataDir(dfsCluster, "data"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@AfterClass
|
@AfterClass
|
||||||
public static void teardownClass() throws Exception {
|
public static void teardownClass() throws Exception {
|
||||||
// there should be no new fails from this test
|
cluster.shutdown(); // need to close before the MiniDFSCluster
|
||||||
assertEquals(0, HdfsUpdateLog.INIT_FAILED_LOGS_COUNT.get() - initialFailLogsCount);
|
|
||||||
HdfsTestUtil.teardownClass(dfsCluster);
|
HdfsTestUtil.teardownClass(dfsCluster);
|
||||||
System.clearProperty("solr.hdfs.blockcache.enabled");
|
|
||||||
dfsCluster = null;
|
dfsCluster = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected String getDataDir(String dataDir) throws IOException {
|
|
||||||
return HdfsTestUtil.getDataDir(dfsCluster, dataDir);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,42 +16,40 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.cloud.hdfs;
|
package org.apache.solr.cloud.hdfs;
|
||||||
|
|
||||||
import java.io.IOException;
|
import com.carrotsearch.randomizedtesting.annotations.ThreadLeakFilters;
|
||||||
|
|
||||||
import org.apache.hadoop.hdfs.MiniDFSCluster;
|
import org.apache.hadoop.hdfs.MiniDFSCluster;
|
||||||
import org.apache.lucene.util.LuceneTestCase.Slow;
|
import org.apache.lucene.util.LuceneTestCase.Slow;
|
||||||
import org.apache.solr.cloud.RecoveryZkTest;
|
import org.apache.solr.cloud.RecoveryZkTest;
|
||||||
|
import org.apache.solr.common.cloud.ZkConfigManager;
|
||||||
import org.apache.solr.util.BadHdfsThreadsFilter;
|
import org.apache.solr.util.BadHdfsThreadsFilter;
|
||||||
import org.junit.AfterClass;
|
import org.junit.AfterClass;
|
||||||
import org.junit.BeforeClass;
|
import org.junit.BeforeClass;
|
||||||
|
|
||||||
import com.carrotsearch.randomizedtesting.annotations.Nightly;
|
|
||||||
import com.carrotsearch.randomizedtesting.annotations.ThreadLeakFilters;
|
|
||||||
|
|
||||||
@Slow
|
@Slow
|
||||||
@Nightly
|
//@Nightly
|
||||||
@ThreadLeakFilters(defaultFilters = true, filters = {
|
@ThreadLeakFilters(defaultFilters = true, filters = {
|
||||||
BadHdfsThreadsFilter.class // hdfs currently leaks thread(s)
|
BadHdfsThreadsFilter.class // hdfs currently leaks thread(s)
|
||||||
})
|
})
|
||||||
public class HdfsRecoveryZkTest extends RecoveryZkTest {
|
public class HdfsRecoveryZkTest extends RecoveryZkTest {
|
||||||
|
|
||||||
private static MiniDFSCluster dfsCluster;
|
private static MiniDFSCluster dfsCluster;
|
||||||
|
|
||||||
@BeforeClass
|
@BeforeClass
|
||||||
public static void setupClass() throws Exception {
|
public static void setupClass() throws Exception {
|
||||||
dfsCluster = HdfsTestUtil.setupClass(createTempDir().toFile().getAbsolutePath());
|
dfsCluster = HdfsTestUtil.setupClass(createTempDir().toFile().getAbsolutePath());
|
||||||
System.setProperty("solr.hdfs.blockcache.blocksperbank", "2048");
|
System.setProperty("solr.hdfs.blockcache.blocksperbank", "2048");
|
||||||
|
|
||||||
|
ZkConfigManager configManager = new ZkConfigManager(zkClient());
|
||||||
|
configManager.uploadConfigDir(configset("cloud-hdfs"), "conf");
|
||||||
|
|
||||||
|
System.setProperty("solr.hdfs.home", HdfsTestUtil.getDataDir(dfsCluster, "data"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@AfterClass
|
@AfterClass
|
||||||
public static void teardownClass() throws Exception {
|
public static void teardownClass() throws Exception {
|
||||||
|
cluster.shutdown(); // need to close before the MiniDFSCluster
|
||||||
HdfsTestUtil.teardownClass(dfsCluster);
|
HdfsTestUtil.teardownClass(dfsCluster);
|
||||||
dfsCluster = null;
|
dfsCluster = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected String getDataDir(String dataDir) throws IOException {
|
|
||||||
return HdfsTestUtil.getDataDir(dfsCluster, dataDir);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -71,7 +71,6 @@ public class BasicAuthStandaloneTest extends AbstractSolrTestCase {
|
||||||
instance = new SolrInstance("inst", null);
|
instance = new SolrInstance("inst", null);
|
||||||
instance.setUp();
|
instance.setUp();
|
||||||
jetty = createJetty(instance);
|
jetty = createJetty(instance);
|
||||||
initCore("solrconfig.xml", "schema.xml", instance.getHomeDir().toString());
|
|
||||||
securityConfHandler = new SecurityConfHandlerLocalForTesting(jetty.getCoreContainer());
|
securityConfHandler = new SecurityConfHandlerLocalForTesting(jetty.getCoreContainer());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -191,6 +191,10 @@ public abstract class CollectionAdminRequest<T extends CollectionAdminResponse>
|
||||||
@Deprecated
|
@Deprecated
|
||||||
public abstract AsyncCollectionSpecificAdminRequest setCollectionName(String collection);
|
public abstract AsyncCollectionSpecificAdminRequest setCollectionName(String collection);
|
||||||
|
|
||||||
|
public String getCollectionName() {
|
||||||
|
return collection;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public SolrParams getParams() {
|
public SolrParams getParams() {
|
||||||
ModifiableSolrParams params = new ModifiableSolrParams(super.getParams());
|
ModifiableSolrParams params = new ModifiableSolrParams(super.getParams());
|
||||||
|
@ -1601,6 +1605,13 @@ public abstract class CollectionAdminRequest<T extends CollectionAdminResponse>
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public AddReplica withProperty(String key, String value) {
|
||||||
|
if (this.properties == null)
|
||||||
|
this.properties = new Properties();
|
||||||
|
this.properties.setProperty(key, value);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
public String getNode() {
|
public String getNode() {
|
||||||
return node;
|
return node;
|
||||||
}
|
}
|
||||||
|
@ -2178,8 +2189,9 @@ public abstract class CollectionAdminRequest<T extends CollectionAdminResponse>
|
||||||
/**
|
/**
|
||||||
* Returns a SolrRequest to get a list of collections in the cluster
|
* Returns a SolrRequest to get a list of collections in the cluster
|
||||||
*/
|
*/
|
||||||
public static List listCollections() {
|
public static java.util.List<String> listCollections(SolrClient client) throws IOException, SolrServerException {
|
||||||
return new List();
|
CollectionAdminResponse resp = new List().process(client);
|
||||||
|
return (java.util.List<String>) resp.getResponse().get("collections");
|
||||||
}
|
}
|
||||||
|
|
||||||
// LIST request
|
// LIST request
|
||||||
|
|
|
@ -620,8 +620,14 @@ public class CoreAdminRequest extends SolrRequest<CoreAdminResponse> {
|
||||||
}
|
}
|
||||||
|
|
||||||
public static CoreStatus getCoreStatus(String coreName, SolrClient client) throws SolrServerException, IOException {
|
public static CoreStatus getCoreStatus(String coreName, SolrClient client) throws SolrServerException, IOException {
|
||||||
|
return getCoreStatus(coreName, true, client);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static CoreStatus getCoreStatus(String coreName, boolean getIndexInfo, SolrClient client)
|
||||||
|
throws SolrServerException, IOException {
|
||||||
CoreAdminRequest req = new CoreAdminRequest();
|
CoreAdminRequest req = new CoreAdminRequest();
|
||||||
req.setAction(CoreAdminAction.STATUS);
|
req.setAction(CoreAdminAction.STATUS);
|
||||||
|
req.setIndexInfoNeeded(getIndexInfo);
|
||||||
return new CoreStatus(req.process(client).getCoreStatus(coreName));
|
return new CoreStatus(req.process(client).getCoreStatus(coreName));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -218,6 +218,13 @@ public class UpdateRequest extends AbstractUpdateRequest {
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public UpdateRequest withRoute(String route) {
|
||||||
|
if (params == null)
|
||||||
|
params = new ModifiableSolrParams();
|
||||||
|
params.set(ROUTE, route);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
public UpdateResponse commit(SolrClient client, String collection) throws IOException, SolrServerException {
|
public UpdateResponse commit(SolrClient client, String collection) throws IOException, SolrServerException {
|
||||||
if (params == null)
|
if (params == null)
|
||||||
params = new ModifiableSolrParams();
|
params = new ModifiableSolrParams();
|
||||||
|
@ -524,4 +531,5 @@ public class UpdateRequest extends AbstractUpdateRequest {
|
||||||
public void lastDocInBatch() {
|
public void lastDocInBatch() {
|
||||||
isLastDocInBatch = true;
|
isLastDocInBatch = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,8 +21,11 @@ import java.util.Collections;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.LinkedHashMap;
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.List;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.function.Predicate;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
import org.noggit.JSONUtil;
|
import org.noggit.JSONUtil;
|
||||||
import org.noggit.JSONWriter;
|
import org.noggit.JSONWriter;
|
||||||
|
@ -218,6 +221,13 @@ public class Slice extends ZkNodeProps implements Iterable<Replica> {
|
||||||
return replicas.values();
|
return replicas.values();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Gets all replicas that match a predicate
|
||||||
|
*/
|
||||||
|
public List<Replica> getReplicas(Predicate<Replica> pred) {
|
||||||
|
return replicas.values().stream().filter(pred).collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the map of coreNodeName to replicas for this slice.
|
* Get the map of coreNodeName to replicas for this slice.
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -88,7 +88,7 @@ public class MiniSolrCloudCluster {
|
||||||
" \n" +
|
" \n" +
|
||||||
"</solr>\n";
|
"</solr>\n";
|
||||||
|
|
||||||
private final ZkTestServer zkServer;
|
private ZkTestServer zkServer; // non-final due to injectChaos()
|
||||||
private final boolean externalZkServer;
|
private final boolean externalZkServer;
|
||||||
private final List<JettySolrRunner> jettys = new CopyOnWriteArrayList<>();
|
private final List<JettySolrRunner> jettys = new CopyOnWriteArrayList<>();
|
||||||
private final Path baseDir;
|
private final Path baseDir;
|
||||||
|
@ -328,6 +328,10 @@ public class MiniSolrCloudCluster {
|
||||||
.build());
|
.build());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public JettySolrRunner getJettySolrRunner(int index) {
|
||||||
|
return jettys.get(index);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Start a new Solr instance on a particular servlet context
|
* Start a new Solr instance on a particular servlet context
|
||||||
*
|
*
|
||||||
|
@ -441,6 +445,10 @@ public class MiniSolrCloudCluster {
|
||||||
return solrClient;
|
return solrClient;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public SolrZkClient getZkClient() {
|
||||||
|
return solrClient.getZkStateReader().getZkClient();
|
||||||
|
}
|
||||||
|
|
||||||
protected CloudSolrClient buildSolrClient() {
|
protected CloudSolrClient buildSolrClient() {
|
||||||
return new Builder()
|
return new Builder()
|
||||||
.withZkHost(getZkServer().getZkAddress())
|
.withZkHost(getZkServer().getZkAddress())
|
||||||
|
@ -497,4 +505,29 @@ public class MiniSolrCloudCluster {
|
||||||
log.info("Expired zookeeper session {} from node {}", sessionId, jetty.getBaseUrl());
|
log.info("Expired zookeeper session {} from node {}", sessionId, jetty.getBaseUrl());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void injectChaos(Random random) throws Exception {
|
||||||
|
|
||||||
|
// sometimes we restart one of the jetty nodes
|
||||||
|
if (random.nextBoolean()) {
|
||||||
|
JettySolrRunner jetty = jettys.get(random.nextInt(jettys.size()));
|
||||||
|
ChaosMonkey.stop(jetty);
|
||||||
|
log.info("============ Restarting jetty");
|
||||||
|
ChaosMonkey.start(jetty);
|
||||||
|
}
|
||||||
|
|
||||||
|
// sometimes we restart zookeeper
|
||||||
|
if (random.nextBoolean()) {
|
||||||
|
zkServer.shutdown();
|
||||||
|
log.info("============ Restarting zookeeper");
|
||||||
|
zkServer = new ZkTestServer(zkServer.getZkDir(), zkServer.getPort());
|
||||||
|
zkServer.run();
|
||||||
|
}
|
||||||
|
|
||||||
|
// sometimes we cause a connection loss - sometimes it will hit the overseer
|
||||||
|
if (random.nextBoolean()) {
|
||||||
|
JettySolrRunner jetty = jettys.get(random.nextInt(jettys.size()));
|
||||||
|
ChaosMonkey.causeConnectionLoss(jetty);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -44,6 +44,7 @@ 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.cloud.SolrZkClient;
|
||||||
|
import org.apache.solr.common.cloud.ZkStateReader;
|
||||||
import org.junit.AfterClass;
|
import org.junit.AfterClass;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
|
|
||||||
|
@ -174,7 +175,10 @@ public class SolrCloudTestCase extends SolrTestCaseJ4 {
|
||||||
/** The cluster */
|
/** The cluster */
|
||||||
protected static MiniSolrCloudCluster cluster;
|
protected static MiniSolrCloudCluster cluster;
|
||||||
|
|
||||||
protected SolrZkClient zkClient() {
|
protected static SolrZkClient zkClient() {
|
||||||
|
ZkStateReader reader = cluster.getSolrClient().getZkStateReader();
|
||||||
|
if (reader == null)
|
||||||
|
cluster.getSolrClient().connect();
|
||||||
return cluster.getSolrClient().getZkStateReader().getZkClient();
|
return cluster.getSolrClient().getZkStateReader().getZkClient();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue