2014-08-06 12:30:01 -04:00
#!/usr/bin/env bash
# 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.
# CONTROLLING STARTUP:
#
# Use solr -help to see available command-line options. In addition
# to passing command-line options, this script looks for an include
# file named solr.in.sh to set environment variables. Specifically,
# the following locations are searched:
#
# ./
# $HOME/.solr.in.sh
# /usr/share/solr
# /usr/local/share/solr
# /opt/solr
#
# Another option is to specify the full path to the include file in the
# environment. For example:
#
# $ SOLR_INCLUDE=/path/to/solr.in.sh solr start
#
# Note: This is particularly handy for running multiple instances on a
# single installation, or for quick tests.
#
# Finally, developers and enthusiasts who frequently run from an SVN
# checkout, and do not want to locally modify bin/solr.in.sh, can put
# a customized include file at ~/.solr.in.sh.
#
# If you would rather configure startup entirely from the environment, you
# can disable the include by exporting an empty SOLR_INCLUDE, or by
# ensuring that no include files exist in the aforementioned search list.
SOLR_SCRIPT="$0"
verbose=false
2014-08-19 22:56:18 -04:00
THIS_OS=`uname -s`
2014-09-05 12:30:23 -04:00
hasLsof=$(which lsof)
2014-10-09 18:28:05 -04:00
stop_all=false
2014-09-05 12:30:23 -04:00
2014-08-19 22:56:18 -04:00
# for now, we don't support running this script from cygwin due to problems
# like not having lsof, ps waux, curl, and awkward directory handling
if [ "${THIS_OS:0:6}" == "CYGWIN" ]; then
echo -e "This script does not support cygwin due to severe limitations and lack of adherence\nto BASH standards, such as lack of lsof, curl, and ps options.\n\nPlease use the native solr.cmd script on Windows!"
exit 1
fi
2014-08-06 12:30:01 -04:00
# Resolve symlinks to this script
while [ -h "$SOLR_SCRIPT" ] ; do
ls=`ls -ld "$SOLR_SCRIPT"`
# Drop everything prior to ->
link=`expr "$ls" : '.*-> \(.*\)$'`
if expr "$link" : '/.*' > /dev/null; then
SOLR_SCRIPT="$link"
else
SOLR_SCRIPT=`dirname "$SOLR_SCRIPT"`/"$link"
fi
done
SOLR_TIP=`dirname "$SOLR_SCRIPT"`/..
SOLR_TIP=`cd "$SOLR_TIP"; pwd`
# TODO: see SOLR-3619, need to support server or example
# depending on the version of Solr
if [ -e "$SOLR_TIP/server/start.jar" ]; then
DEFAULT_SERVER_DIR=$SOLR_TIP/server
else
DEFAULT_SERVER_DIR=$SOLR_TIP/example
fi
2014-10-09 14:42:21 -04:00
# If an include wasn't specified in the environment, then search for one...
if [ "x$SOLR_INCLUDE" == "x" ]; then
# Locations (in order) to use when searching for an include file.
for include in "`dirname "$0"`/solr.in.sh" \
"$HOME/.solr.in.sh" \
/usr/share/solr/solr.in.sh \
/usr/local/share/solr/solr.in.sh \
/opt/solr/solr.in.sh; do
if [ -r "$include" ]; then
. "$include"
break
fi
done
elif [ -r "$SOLR_INCLUDE" ]; then
. "$SOLR_INCLUDE"
fi
2014-08-06 12:30:01 -04:00
if [ "$SOLR_JAVA_HOME" != "" ]; then
JAVA=$SOLR_JAVA_HOME/bin/java
elif [ -n "$JAVA_HOME" ]; then
for java in "$JAVA_HOME"/bin/amd64/java "$JAVA_HOME"/bin/java; do
if [ -x "$java" ]; then
JAVA="$java"
break
fi
done
else
JAVA=java
fi
# test that Java exists and is executable on this server
$JAVA -version >/dev/null 2>&1 || { echo >&2 "Java is required to run Solr! Please install Java 7 or 8 before running this script."; exit 1; }
function print_usage() {
CMD="$1"
ERROR_MSG="$2"
if [ "$ERROR_MSG" != "" ]; then
2014-08-19 22:56:18 -04:00
echo -e "\nERROR: $ERROR_MSG\n"
2014-08-06 12:30:01 -04:00
fi
if [ "$CMD" == "" ]; then
echo ""
echo "Usage: solr COMMAND OPTIONS"
echo " where COMMAND is one of: start, stop, restart, healthcheck"
echo ""
2014-10-09 14:42:21 -04:00
echo " Standalone server example (start Solr running in the background on port 8984):"
2014-08-06 12:30:01 -04:00
echo ""
echo " ./solr start -p 8984"
echo ""
2014-10-09 14:42:21 -04:00
echo " SolrCloud example (start Solr running in SolrCloud mode using localhost:2181 to connect to ZooKeeper, with 1g max heap size and remote Java debug options enabled):"
echo ""
echo " ./solr start -c -m 1g -z localhost:2181 -a \"-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=n,address=1044\""
echo ""
2014-08-06 12:30:01 -04:00
echo "Pass -help after any COMMAND to see command-specific usage information,"
2014-10-09 14:42:21 -04:00
echo " such as: ./solr start -help or ./solr stop -help"
2014-08-06 12:30:01 -04:00
echo ""
elif [[ "$CMD" == "start" || "$CMD" == "restart" ]]; then
echo ""
2014-10-09 14:42:21 -04:00
echo "Usage: solr $CMD [-f] [-c] [-h hostname] [-p port] [-d directory] [-z zkHost] [-m memory] [-e example] [-s solr.solr.home] [-a \"additional-options\"] [-V]"
2014-08-06 12:30:01 -04:00
echo ""
echo " -f Start Solr in foreground; default starts Solr in the background"
echo " and sends stdout / stderr to solr-PORT-console.log"
echo ""
echo " -c or -cloud Start Solr in SolrCloud mode; if -z not supplied, an embedded ZooKeeper"
echo " instance is started on Solr port+1000, such as 9983 if Solr is bound to 8983"
echo ""
echo " -h <host> Specify the hostname for this Solr instance"
echo ""
echo " -p <port> Specify the port to start the Solr HTTP listener on; default is 8983"
echo ""
echo " -d <dir> Specify the Solr server directory; defaults to server"
echo ""
echo " -z <zkHost> ZooKeeper connection string; only used when running in SolrCloud mode using -c"
echo " To launch an embedded ZooKeeper instance, don't pass this parameter."
echo ""
echo " -m <memory> Sets the min (-Xms) and max (-Xmx) heap size for the JVM, such as: -m 4g"
echo " results in: -Xms4g -Xmx4g; by default, this script sets the heap size to 512m"
echo ""
2014-10-09 14:42:21 -04:00
echo " -s <dir> Sets the solr.solr.home system property; Solr will create core directories under"
echo " this directory. This allows you to run multiple Solr instances on the same host"
echo " while reusing the same server directory set using the -d parameter. If set, the"
echo " specified directory should contain a solr.xml file. The default value is example/solr."
echo " This parameter is ignored when running examples (-e), as the solr.solr.home depends"
echo " on which example is run."
echo ""
2014-08-06 12:30:01 -04:00
echo " -e <example> Name of the example to run; available examples:"
echo " cloud: SolrCloud example"
echo " default: Solr default example"
echo " dih: Data Import Handler"
echo " schemaless: Schema-less example"
echo " multicore: Multicore"
echo ""
2014-09-22 12:48:32 -04:00
echo " -a Additional parameters to pass to the JVM when starting Solr, such as to setup"
echo " Java debug options. For example, to enable a Java debugger to attach to the Solr JVM"
echo " you could pass: -a \"-agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=18983\""
echo " In most cases, you should wrap the additional parameters in double quotes."
echo ""
2014-08-06 12:30:01 -04:00
echo " -noprompt Don't prompt for input; accept all defaults when running examples that accept user input"
echo ""
echo " -V Verbose messages from this script"
echo ""
elif [ "$CMD" == "stop" ]; then
echo ""
2014-09-22 12:48:32 -04:00
echo "Usage: solr stop [-k key] [-p port] [-V]"
2014-08-06 12:30:01 -04:00
echo ""
echo " -k <key> Stop key; default is solrrocks"
echo ""
2014-08-19 22:56:18 -04:00
echo " -p <port> Specify the port the Solr HTTP listener is bound to; default is 8983"
2014-08-06 12:30:01 -04:00
echo ""
2014-10-09 18:28:05 -04:00
echo " -all Find and stop all running Solr servers on this host"
echo ""
2014-08-06 12:30:01 -04:00
echo " -V Verbose messages from this script"
echo ""
echo "NOTE: If port is not specified, then all running Solr servers are stopped."
echo ""
elif [ "$CMD" == "healthcheck" ]; then
echo ""
echo "Usage: solr healthcheck [-c collection] [-z zkHost]"
echo ""
echo " -c <collection> Collection to run healthcheck against."
echo ""
echo " -z <zkHost> ZooKeeper connection string; default is localhost:9983"
echo ""
fi
} # end print_usage
# used to show the script is still alive when waiting on work to complete
spinner()
{
local pid=$1
local delay=0.5
local spinstr='|/-\'
while [ "$(ps a | awk '{print $1}' | grep $pid)" ]; do
local temp=${spinstr#?}
printf " [%c] " "$spinstr"
local spinstr=$temp${spinstr%"$temp"}
sleep $delay
printf "\b\b\b\b\b\b"
done
printf " \b\b\b\b"
}
2014-10-09 18:28:05 -04:00
# given a port, find the pid for a Solr process
function solr_pid_by_port() {
THE_PORT="$1"
if [ -e "$SOLR_TIP/bin/solr-$THE_PORT.pid" ]; then
PID=`cat $SOLR_TIP/bin/solr-$THE_PORT.pid`
CHECK_PID=`ps waux | awk '{print $2}' | grep $PID | sort -r | tr -d ' '`
if [ "$CHECK_PID" != "" ]; then
local solrPID=$PID
fi
fi
echo "$solrPID"
}
2014-08-06 12:30:01 -04:00
# extract the value of the -Djetty.port parameter from a running Solr process
function jetty_port() {
SOLR_PID="$1"
SOLR_PROC=`ps waux | grep $SOLR_PID | grep start.jar | grep jetty.port`
IFS=' ' read -a proc_args <<< "$SOLR_PROC"
for arg in "${proc_args[@]}"
do
IFS='=' read -a pair <<< "$arg"
if [ "${pair[0]}" == "-Djetty.port" ]; then
local jetty_port="${pair[1]}"
break
fi
done
echo "$jetty_port"
} # end jetty_port func
# run a Solr command-line tool using the SolrCLI class;
# useful for doing cross-platform work from the command-line using Java
function run_tool() {
# Extract the solr.war if it hasn't been done already (so we can access the SolrCLI class)
2014-10-09 14:42:21 -04:00
if [[ -e $DEFAULT_SERVER_DIR/webapps/solr.war && ! -d "$DEFAULT_SERVER_DIR/solr-webapp/webapp" ]]; then
2014-08-06 12:30:01 -04:00
(mkdir -p $DEFAULT_SERVER_DIR/solr-webapp/webapp && cd $DEFAULT_SERVER_DIR/solr-webapp/webapp && jar xf $DEFAULT_SERVER_DIR/webapps/solr.war)
fi
"$JAVA" -Dlog4j.configuration=file:$DEFAULT_SERVER_DIR/scripts/cloud-scripts/log4j.properties \
-classpath "$DEFAULT_SERVER_DIR/solr-webapp/webapp/WEB-INF/lib/*:$DEFAULT_SERVER_DIR/lib/ext/*" \
org.apache.solr.util.SolrCLI $*
} # end run_tool function
# get information about any Solr nodes running on this host
function get_info() {
# first, see if Solr is running
2014-10-09 18:28:05 -04:00
numSolrs=`find $SOLR_TIP/bin -name "solr-*.pid" -type f | wc -l | tr -d ' '`
2014-08-06 12:30:01 -04:00
if [ "$numSolrs" != "0" ]; then
echo -e "\nFound $numSolrs Solr nodes: "
2014-10-09 18:28:05 -04:00
for PIDF in `find $SOLR_TIP/bin -name "solr-*.pid" -type f`
2014-08-06 12:30:01 -04:00
do
2014-10-09 18:28:05 -04:00
ID=`cat $PIDF`
2014-08-06 12:30:01 -04:00
port=`jetty_port "$ID"`
if [ "$port" != "" ]; then
echo ""
echo "Found Solr process $ID running on port $port"
run_tool status -solr http://localhost:$port/solr
echo ""
fi
done
else
2014-10-13 15:23:22 -04:00
# no pid files but check using ps just to be sure
numSolrs=`ps waux | grep java | grep start.jar | wc -l | sed -e 's/^[ \t]*//'`
if [ "$numSolrs" != "0" ]; then
echo -e "\nFound $numSolrs Solr nodes: "
for ID in `ps waux | grep java | grep start.jar | awk '{print $2}' | sort -r`
do
port=`jetty_port "$ID"`
if [ "$port" != "" ]; then
echo ""
echo "Found Solr process $ID running on port $port"
run_tool status -solr http://localhost:$port/solr
echo ""
fi
done
else
echo -e "\nNo Solr nodes are running.\n"
fi
2014-08-06 12:30:01 -04:00
fi
} # end get_info
# tries to gracefully stop Solr using the Jetty
# stop command and if that fails, then uses kill -9
function stop_solr() {
DIR="$1"
SOLR_PORT="$2"
STOP_PORT="79${SOLR_PORT: -2}"
STOP_KEY="$3"
2014-10-09 18:28:05 -04:00
SOLR_PID="$4"
2014-08-06 12:30:01 -04:00
if [ "$SOLR_PID" != "" ]; then
2014-10-23 17:45:21 -04:00
echo -e "Sending stop command to Solr running on port $SOLR_PORT ... waiting 5 seconds to allow Jetty process $SOLR_PID to stop gracefully."
2014-08-06 12:30:01 -04:00
$JAVA -jar $DIR/start.jar STOP.PORT=$STOP_PORT STOP.KEY=$STOP_KEY --stop || true
2014-09-05 12:30:23 -04:00
(sleep 5) &
spinner $!
2014-10-09 18:28:05 -04:00
rm -f $SOLR_TIP/bin/solr-$SOLR_PORT.pid
2014-08-06 12:30:01 -04:00
else
echo -e "No Solr nodes found to stop."
exit 0
fi
2014-10-09 18:28:05 -04:00
CHECK_PID=`ps waux | awk '{print $2}' | grep $SOLR_PID | sort -r | tr -d ' '`
if [ "$CHECK_PID" != "" ]; then
2014-08-06 12:30:01 -04:00
echo -e "Solr process $SOLR_PID is still running; forcefully killing it now."
kill -9 $SOLR_PID
echo "Killed process $SOLR_PID"
2014-10-09 18:28:05 -04:00
rm -f $SOLR_TIP/bin/solr-$SOLR_PORT.pid
sleep 1
2014-08-06 12:30:01 -04:00
fi
2014-10-09 18:28:05 -04:00
CHECK_PID=`ps waux | awk '{print $2}' | grep $SOLR_PID | sort -r | tr -d ' '`
if [ "$CHECK_PID" != "" ]; then
2014-08-06 12:30:01 -04:00
echo "ERROR: Failed to kill previous Solr Java process $SOLR_PID ... script fails."
exit 1
fi
} # end stop_solr
if [ $# -eq 1 ]; then
case $1 in
-help|-usage)
print_usage ""
exit
;;
-info|-i)
get_info
exit
;;
esac
fi
if [ $# -gt 0 ]; then
# if first arg starts with a dash (and it's not -help or -info),
# then assume they are starting Solr, such as: solr -f
if [[ $1 == -* ]]; then
SCRIPT_CMD="start"
else
SCRIPT_CMD=$1
shift
fi
else
# no args - just show usage and exit
print_usage ""
exit
fi
# run a healthcheck and exit if requested
if [ "$SCRIPT_CMD" == "healthcheck" ]; then
if [ $# -gt 0 ]; then
while true; do
case $1 in
-c|-collection)
2014-08-19 22:56:18 -04:00
if [ "${2:0:1}" == "-" ]; then
2014-08-21 11:57:33 -04:00
print_usage "$SCRIPT_CMD" "Expected collection name but found $2 instead!"
2014-08-19 22:56:18 -04:00
exit 1
fi
2014-08-06 12:30:01 -04:00
HEALTHCHECK_COLLECTION=$2
shift 2
;;
2014-08-19 22:56:18 -04:00
-z|-zkhost)
if [ "${2:0:1}" == "-" ]; then
2014-08-21 11:57:33 -04:00
print_usage "$SCRIPT_CMD" "Expected a ZooKeeper connection string but found $2 instead!"
2014-08-19 22:56:18 -04:00
exit 1
fi
2014-08-06 12:30:01 -04:00
ZK_HOST="$2"
shift 2
;;
-help|-usage)
2014-08-19 22:56:18 -04:00
print_usage "$SCRIPT_CMD"
2014-08-06 12:30:01 -04:00
exit 0
;;
--)
shift
break
;;
*)
if [ "$1" != "" ]; then
2014-08-19 22:56:18 -04:00
print_usage "$SCRIPT_CMD" "Unrecognized or misplaced argument: $1!"
2014-08-06 12:30:01 -04:00
exit 1
else
break # out-of-args, stop looping
fi
;;
esac
done
fi
if [ "$ZK_HOST" == "" ]; then
ZK_HOST=localhost:9983
fi
2014-08-19 22:56:18 -04:00
if [ "$HEALTHCHECK_COLLECTION" == "" ]; then
echo "collection parameter is required!"
print_usage "healthcheck"
exit 1
fi
2014-08-06 12:30:01 -04:00
run_tool healthcheck -zkHost $ZK_HOST -collection $HEALTHCHECK_COLLECTION
exit $?
fi
# verify the command given is supported
if [ "$SCRIPT_CMD" != "stop" ] && [ "$SCRIPT_CMD" != "start" ] && [ "$SCRIPT_CMD" != "restart" ]; then
print_usage "" "$SCRIPT_CMD not supported!"
exit 1
fi
# Run in foreground (default is to run in the background)
FG="false"
noprompt=false
if [ $# -gt 0 ]; then
while true; do
case $1 in
-c|-cloud)
SOLR_MODE="solrcloud"
shift
;;
-d|-dir)
2014-08-19 22:56:18 -04:00
if [ "${2:0:1}" == "-" ]; then
2014-08-21 11:57:33 -04:00
print_usage "$SCRIPT_CMD" "Expected directory but found $2 instead!"
2014-08-19 22:56:18 -04:00
exit 1
fi
2014-08-06 12:30:01 -04:00
# see if the arg value is relative to the tip vs full path
2014-10-09 14:42:21 -04:00
if [[ $2 != /* ]] && [[ -d "$SOLR_TIP/$2" ]]; then
SOLR_SERVER_DIR="$SOLR_TIP/$2"
else
2014-08-06 12:30:01 -04:00
SOLR_SERVER_DIR="$2"
fi
shift 2
;;
2014-10-09 14:42:21 -04:00
-s|-solr.home)
if [ "${2:0:1}" == "-" ]; then
print_usage "$SCRIPT_CMD" "Expected directory but found $2 instead!"
exit 1
fi
SOLR_HOME="$2"
shift 2
;;
2014-08-06 12:30:01 -04:00
-e|-example)
2014-08-19 22:56:18 -04:00
if [ "${2:0:1}" == "-" ]; then
2014-08-21 11:57:33 -04:00
print_usage "$SCRIPT_CMD" "Expected example name but found $2 instead!"
2014-08-19 22:56:18 -04:00
exit 1
fi
2014-08-06 12:30:01 -04:00
EXAMPLE="$2"
shift 2
;;
-f|-foreground)
FG="true"
shift
;;
-h|-host)
2014-08-19 22:56:18 -04:00
if [ "${2:0:1}" == "-" ]; then
2014-08-21 11:57:33 -04:00
print_usage "$SCRIPT_CMD" "Expected hostname but found $2 instead!"
2014-08-19 22:56:18 -04:00
exit 1
fi
2014-08-06 12:30:01 -04:00
SOLR_HOST="$2"
shift 2
;;
-m|-memory)
2014-08-19 22:56:18 -04:00
if [ "${2:0:1}" == "-" ]; then
2014-08-21 11:57:33 -04:00
print_usage "$SCRIPT_CMD" "Expected memory setting but found $2 instead!"
2014-08-19 22:56:18 -04:00
exit 1
fi
2014-08-06 12:30:01 -04:00
SOLR_HEAP="$2"
shift 2
;;
-p|-port)
2014-08-19 22:56:18 -04:00
if [ "${2:0:1}" == "-" ]; then
2014-08-21 11:57:33 -04:00
print_usage "$SCRIPT_CMD" "Expected port number but found $2 instead!"
2014-08-19 22:56:18 -04:00
exit 1
fi
2014-08-06 12:30:01 -04:00
SOLR_PORT="$2"
shift 2
;;
-z|-zkhost)
2014-08-19 22:56:18 -04:00
if [ "${2:0:1}" == "-" ]; then
2014-08-21 11:57:33 -04:00
print_usage "$SCRIPT_CMD" "Expected ZooKeeper connection string but found $2 instead!"
2014-08-19 22:56:18 -04:00
exit 1
fi
2014-08-06 12:30:01 -04:00
ZK_HOST="$2"
shift 2
;;
2014-09-22 12:48:32 -04:00
-a|-addlopts)
ADDITIONAL_CMD_OPTS="$2"
shift 2
;;
2014-10-09 14:42:21 -04:00
-k|-key)
STOP_KEY="$2"
shift 2
;;
2014-08-06 12:30:01 -04:00
-help|-usage)
2014-08-19 22:56:18 -04:00
print_usage "$SCRIPT_CMD"
2014-08-06 12:30:01 -04:00
exit 0
;;
-noprompt)
noprompt=true
shift
;;
-V|-verbose)
verbose=true
shift
;;
2014-10-09 18:28:05 -04:00
-all)
stop_all=true
shift
;;
2014-08-06 12:30:01 -04:00
--)
shift
break
;;
*)
if [ "$1" != "" ]; then
2014-08-19 22:56:18 -04:00
print_usage "$SCRIPT_CMD" "Error parsing argument $1!"
2014-08-06 12:30:01 -04:00
exit 1
else
break # out-of-args, stop looping
fi
;;
esac
done
fi
if $verbose ; then
echo "Using Solr root directory: $SOLR_TIP"
echo "Using Java: $JAVA"
$JAVA -version
fi
if [ "$SOLR_HOST" != "" ]; then
SOLR_HOST_ARG="-Dhost=$SOLR_HOST"
else
SOLR_HOST_ARG=""
fi
if [ "$SOLR_SERVER_DIR" == "" ]; then
SOLR_SERVER_DIR=$DEFAULT_SERVER_DIR
fi
if [ ! -e "$SOLR_SERVER_DIR" ]; then
echo -e "\nSolr server directory $SOLR_SERVER_DIR not found!\n"
exit 1
fi
CLOUD_NUM_NODES=2
declare -a CLOUD_PORTS=('8983' '7574' '8984' '7575');
# select solr.solr.home based on the desired example
if [ "$EXAMPLE" != "" ]; then
case $EXAMPLE in
cloud)
#
# Engage in an interactive session with user to setup the SolrCloud example
#
echo -e "\nWelcome to the SolrCloud example!\n\n"
if $noprompt ; then
CLOUD_NUM_NODES=2
echo -e "Starting up $CLOUD_NUM_NODES Solr nodes for your example SolrCloud cluster."
else
echo -e "This interactive session will help you launch a SolrCloud cluster on your local workstation.\n"
read -e -p "To begin, how many Solr nodes would you like to run in your local cluster? (specify 1-4 nodes) [2] " USER_INPUT
while true
do
CLOUD_NUM_NODES=`echo $USER_INPUT | tr -d ' '`
if [ "$CLOUD_NUM_NODES" == "" ]; then
CLOUD_NUM_NODES=2
fi
if [[ $CLOUD_NUM_NODES > 4 || $CLOUD_NUM_NODES < 1 ]]; then
read -e -p "Please provide a node count between 1 and 4 [2] " USER_INPUT
else
break;
fi
done
echo -e "Ok, let's start up $CLOUD_NUM_NODES Solr nodes for your example SolrCloud cluster.\n"
for (( s=0; s<$CLOUD_NUM_NODES; s++ ))
do
read -e -p "Please enter the port for node$[$s+1] [${CLOUD_PORTS[$s]}] " USER_INPUT
while true
do
# trim whitespace out of the user input
CLOUD_PORT=`echo $USER_INPUT | tr -d ' '`
# handle the default selection or empty input
if [ "$CLOUD_PORT" == "" ]; then
CLOUD_PORT=${CLOUD_PORTS[$s]}
fi
# check to see if something is already bound to that port
2014-09-05 12:30:23 -04:00
if [ "$hasLsof" != "" ]; then
2014-10-23 17:45:21 -04:00
PORT_IN_USE=`lsof -Pni:$CLOUD_PORT`
2014-09-05 12:30:23 -04:00
if [ "$PORT_IN_USE" != "" ]; then
read -e -p "Oops! Looks like port $CLOUD_PORT is already being used by another process. Please choose a different port. " USER_INPUT
else
CLOUD_PORTS[$s]=$CLOUD_PORT
echo $CLOUD_PORT
break;
fi
2014-08-06 12:30:01 -04:00
else
CLOUD_PORTS[$s]=$CLOUD_PORT
echo $CLOUD_PORT
2014-09-05 12:30:23 -04:00
break;
2014-08-06 12:30:01 -04:00
fi
done
done
fi
for (( s=0; s<$CLOUD_NUM_NODES; s++ ))
do
ndx=$[$s+1]
if [ ! -d "$SOLR_TIP/node$ndx" ]; then
echo "Cloning $DEFAULT_SERVER_DIR into $SOLR_TIP/node$ndx"
cp -r $DEFAULT_SERVER_DIR $SOLR_TIP/node$ndx
rm -rf $SOLR_TIP/node$ndx/solr/zoo_data
fi
done
SOLR_MODE="solrcloud"
SOLR_SERVER_DIR="$SOLR_TIP/node1"
SOLR_HOME="$SOLR_SERVER_DIR/solr"
2014-08-21 17:17:38 -04:00
SOLR_PORT=${CLOUD_PORTS[0]}
2014-08-06 12:30:01 -04:00
shift
;;
default)
SOLR_HOME="$SOLR_TIP/example/solr"
shift
;;
dih)
SOLR_HOME="$SOLR_TIP/example/example-DIH/solr"
shift
;;
schemaless)
SOLR_HOME="$SOLR_TIP/example/example-schemaless/solr"
shift
;;
multicore)
SOLR_HOME="$SOLR_TIP/example/multicore"
shift
;;
*)
print_usage "start" "Unsupported example $EXAMPLE! Please choose one of: cloud, dih, schemaless, multicore, or default"
exit 1
;;
esac
fi
if [ "$STOP_KEY" == "" ]; then
STOP_KEY="solrrocks"
fi
# stop all if no port specified
if [[ "$SCRIPT_CMD" == "stop" && "$SOLR_PORT" == "" ]]; then
2014-10-09 18:28:05 -04:00
if $stop_all; then
none_stopped=true
for PIDF in `find $SOLR_TIP/bin -name "solr-*.pid" -type f`
2014-08-06 12:30:01 -04:00
do
2014-10-09 18:28:05 -04:00
NEXT_PID=`cat $PIDF`
port=`jetty_port "$NEXT_PID"`
if [ "$port" != "" ]; then
stop_solr "$SOLR_SERVER_DIR" "$port" "$STOP_KEY" "$NEXT_PID"
none_stopped=false
fi
rm -f $PIDF
done
if $none_stopped; then
echo -e "\nNo Solr nodes found to stop.\n"
fi
2014-08-06 12:30:01 -04:00
else
2014-10-09 18:28:05 -04:00
echo -e "\nERROR: Must either specify a port using -p or -all to stop all Solr nodes on this host.\n"
exit 1
2014-08-06 12:30:01 -04:00
fi
2014-10-09 18:28:05 -04:00
exit
2014-08-06 12:30:01 -04:00
fi
if [ "$SOLR_PORT" == "" ]; then
SOLR_PORT="8983"
fi
if [ "$STOP_PORT" == "" ]; then
STOP_PORT="79${SOLR_PORT: -2}"
fi
if [[ "$SCRIPT_CMD" == "start" ]]; then
# see if Solr is already running
2014-10-09 18:28:05 -04:00
SOLR_PID=`solr_pid_by_port "$SOLR_PORT"`
2014-10-13 12:38:49 -04:00
if [ "$SOLR_PID" == "" ]; then
# not found using the pid file ... but use ps to ensure not found
SOLR_PID=`ps waux | grep start.jar | grep $SOLR_PORT | grep -v grep | awk '{print $2}' | sort -r`
fi
2014-08-06 12:30:01 -04:00
if [ "$SOLR_PID" != "" ]; then
echo -e "\nSolr already running on port $SOLR_PORT (pid: $SOLR_PID)!"
echo -e "Please use the 'restart' command if you want to restart this node.\n"
exit
fi
else
# either stop or restart
2014-10-13 15:23:22 -04:00
# see if Solr is already running
2014-10-09 18:28:05 -04:00
SOLR_PID=`solr_pid_by_port "$SOLR_PORT"`
2014-10-13 15:23:22 -04:00
if [ "$SOLR_PID" == "" ]; then
# not found using the pid file ... but use ps to ensure not found
SOLR_PID=`ps waux | grep start.jar | grep $SOLR_PORT | grep -v grep | awk '{print $2}' | sort -r`
fi
2014-10-09 18:28:05 -04:00
if [ "$SOLR_PID" != "" ]; then
2014-10-13 15:23:22 -04:00
stop_solr "$SOLR_SERVER_DIR" "$SOLR_PORT" "$STOP_KEY" "$SOLR_PID"
2014-10-09 18:28:05 -04:00
else
echo -e "No process found for Solr node running on port $SOLR_PORT"
fi
2014-08-06 12:30:01 -04:00
fi
# backup the log files
if [ -f $SOLR_SERVER_DIR/logs/solr.log ]; then
if $verbose ; then
echo "Backing up $SOLR_SERVER_DIR/logs/solr.log"
fi
mv $SOLR_SERVER_DIR/logs/solr.log $SOLR_SERVER_DIR/logs/solr_log_`date +"%Y%m%d_%H%M"`
fi
if [ -f $SOLR_SERVER_DIR/logs/solr_gc.log ]; then
if $verbose ; then
echo "Backing up $SOLR_SERVER_DIR/logs/solr_gc.log"
fi
mv $SOLR_SERVER_DIR/logs/solr_gc.log $SOLR_SERVER_DIR/logs/solr_gc_log_`date +"%Y%m%d_%H%M"`
fi
if [ "$SCRIPT_CMD" == "stop" ]; then
# already stopped, script is done.
exit 0
fi
# if we get here, then we're starting a new node up ...
2014-10-09 18:28:05 -04:00
if [ "$SOLR_HOME" == "" ]; then
SOLR_HOME="$SOLR_SERVER_DIR/solr"
else
if [[ $SOLR_HOME != /* ]] && [[ -d "$SOLR_SERVER_DIR/$SOLR_HOME" ]]; then
SOLR_HOME="$SOLR_SERVER_DIR/$SOLR_HOME"
fi
fi
if [ ! -e "$SOLR_HOME" ]; then
echo -e "\nSolr home directory $SOLR_HOME not found!\n"
exit 1
fi
if [ ! -e "$SOLR_HOME/solr.xml" ]; then
echo -e "\nSolr home directory $SOLR_HOME must contain a solr.xml file!\n"
exit 1
fi
2014-08-06 12:30:01 -04:00
# if verbose gc logging enabled, setup the location of the log file
if [ "$GC_LOG_OPTS" != "" ]; then
GC_LOG_OPTS="$GC_LOG_OPTS -Xloggc:$SOLR_SERVER_DIR/logs/solr_gc.log"
fi
if [ "$SOLR_MODE" == "solrcloud" ]; then
if [ "$ZK_CLIENT_TIMEOUT" == "" ]; then
ZK_CLIENT_TIMEOUT="15000"
fi
CLOUD_MODE_OPTS="-DzkClientTimeout=$ZK_CLIENT_TIMEOUT"
if [ "$ZK_HOST" != "" ]; then
CLOUD_MODE_OPTS="$CLOUD_MODE_OPTS -DzkHost=$ZK_HOST"
else
if $verbose ; then
echo "Configuring SolrCloud to launch an embedded ZooKeeper using -DzkRun"
fi
CLOUD_MODE_OPTS="$CLOUD_MODE_OPTS -DzkRun"
2014-09-22 13:23:14 -04:00
fi
# and if collection1 needs to be bootstrapped
if [ -e "$SOLR_HOME/collection1/core.properties" ]; then
CLOUD_MODE_OPTS="$CLOUD_MODE_OPTS -Dbootstrap_confdir=./solr/collection1/conf -Dcollection.configName=myconf -DnumShards=1"
2014-08-06 12:30:01 -04:00
fi
2014-08-19 22:56:18 -04:00
2014-08-06 12:30:01 -04:00
fi
# These are useful for attaching remove profilers like VisualVM/JConsole
if [ "$ENABLE_REMOTE_JMX_OPTS" == "true" ]; then
REMOTE_JMX_OPTS="-Dcom.sun.management.jmxremote \
-Dcom.sun.management.jmxremote.local.only=false \
-Dcom.sun.management.jmxremote.ssl=false \
-Dcom.sun.management.jmxremote.authenticate=false \
-Dcom.sun.management.jmxremote.port=10${SOLR_PORT: -2} \
-Dcom.sun.management.jmxremote.rmi.port=10${SOLR_PORT: -2}"
# if the host is set, then set that as the rmi server hostname
if [ "$SOLR_HOST" != "" ]; then
REMOTE_JMX_OPTS="$REMOTE_JMX_OPTS -Djava.rmi.server.hostname=$SOLR_HOST"
fi
else
REMOTE_JMX_OPTS=""
fi
if [ "$SOLR_HEAP" != "" ]; then
SOLR_JAVA_MEM="-Xms$SOLR_HEAP -Xmx$SOLR_HEAP"
fi
if [ "$SOLR_JAVA_MEM" == "" ]; then
SOLR_JAVA_MEM="-Xms512m -Xmx512m"
fi
if [ "$SOLR_TIMEZONE" == "" ]; then
SOLR_TIMEZONE="UTC"
fi
# Launches Solr in foreground/background depending on parameters
function launch_solr() {
run_in_foreground="$1"
stop_port="79${SOLR_PORT: -2}"
SOLR_ADDL_ARGS="$2"
# commented out debugging info
if $verbose ; then
echo -e "\nStarting Solr using the following settings:"
echo -e " SOLR_SERVER_DIR = $SOLR_SERVER_DIR"
echo -e " SOLR_HOME = $SOLR_HOME"
echo -e " JAVA = $JAVA"
echo -e " SOLR_HOST = $SOLR_HOST"
echo -e " SOLR_PORT = $SOLR_PORT"
echo -e " GC_TUNE = $GC_TUNE"
echo -e " GC_LOG_OPTS = $GC_LOG_OPTS"
echo -e " SOLR_JAVA_MEM = $SOLR_JAVA_MEM"
echo -e " REMOTE_JMX_OPTS = $REMOTE_JMX_OPTS"
echo -e " CLOUD_MODE_OPTS = $CLOUD_MODE_OPTS"
echo -e " SOLR_TIMEZONE = $SOLR_TIMEZONE"
if [ "$SOLR_ADDL_ARGS" != "" ]; then
echo -e " SOLR_ADDL_ARGS = $SOLR_ADDL_ARGS"
fi
fi
# need to launch solr from the server dir
cd $SOLR_SERVER_DIR
if [ ! -e "$SOLR_SERVER_DIR/start.jar" ]; then
echo -e "\nERROR: start.jar file not found in $SOLR_SERVER_DIR!\nPlease check your -d parameter to set the correct Solr server directory.\n"
exit 1
fi
SOLR_START_OPTS="-server -Xss256k $SOLR_JAVA_MEM $GC_TUNE $GC_LOG_OPTS $REMOTE_JMX_OPTS \
$CLOUD_MODE_OPTS \
-DSTOP.PORT=$stop_port -DSTOP.KEY=$STOP_KEY \
$SOLR_HOST_ARG -Djetty.port=$SOLR_PORT \
-Dsolr.solr.home=$SOLR_HOME \
2014-10-27 23:05:19 -04:00
-Dsolr.install.dir=$SOLR_TIP \
2014-08-06 12:30:01 -04:00
-Duser.timezone=$SOLR_TIMEZONE \
-Djava.net.preferIPv4Stack=true -Dsolr.autoSoftCommit.maxTime=3000"
if [ "$SOLR_MODE" == "solrcloud" ]; then
IN_CLOUD_MODE=" in SolrCloud mode"
fi
if [ "$run_in_foreground" == "true" ]; then
echo -e "\nStarting Solr$IN_CLOUD_MODE on port $SOLR_PORT from $SOLR_SERVER_DIR\n"
$JAVA $SOLR_START_OPTS $SOLR_ADDL_ARGS -XX:OnOutOfMemoryError="$SOLR_TIP/bin/oom_solr.sh $SOLR_PORT" -jar start.jar
else
# run Solr in the background
2014-10-09 18:28:05 -04:00
nohup $JAVA $SOLR_START_OPTS $SOLR_ADDL_ARGS -XX:OnOutOfMemoryError="$SOLR_TIP/bin/oom_solr.sh $SOLR_PORT" -jar start.jar 1>$SOLR_SERVER_DIR/logs/solr-$SOLR_PORT-console.log 2>&1 & echo $! > $SOLR_TIP/bin/solr-$SOLR_PORT.pid
2014-08-06 12:30:01 -04:00
2014-08-19 22:56:18 -04:00
# no lsof on cygwin though
2014-09-05 12:30:23 -04:00
if [ "$hasLsof" != "" ]; then
echo -n "Waiting to see Solr listening on port $SOLR_PORT"
# Launch in a subshell to show the spinner
(loops=0
while true
do
2014-10-23 17:45:21 -04:00
running=`lsof -Pni:$SOLR_PORT`
2014-09-05 12:30:23 -04:00
if [ "$running" == "" ]; then
if [ $loops -lt 6 ]; then
sleep 5
loops=$[$loops+1]
else
echo -e "Still not seeing Solr listening on $SOLR_PORT after 30 seconds!"
2014-10-23 17:45:21 -04:00
tail -30 $SOLR_SERVER_DIR/logs/solr.log
2014-09-05 12:30:23 -04:00
exit;
fi
2014-08-06 12:30:01 -04:00
else
2014-09-05 12:30:23 -04:00
SOLR_PID=`ps waux | grep start.jar | grep $SOLR_PORT | grep -v grep | awk '{print $2}' | sort -r`
echo -e "\nStarted Solr server on port $SOLR_PORT (pid=$SOLR_PID). Happy searching!\n"
2014-08-06 12:30:01 -04:00
exit;
2014-09-05 12:30:23 -04:00
fi
done) &
spinner $!
else
SOLR_PID=`ps waux | grep start.jar | grep $SOLR_PORT | grep -v grep | awk '{print $2}' | sort -r`
echo -e "\nStarted Solr server on port $SOLR_PORT (pid=$SOLR_PID). Happy searching!\n"
exit;
fi
2014-08-06 12:30:01 -04:00
fi
}
if [ "$EXAMPLE" != "cloud" ]; then
2014-09-22 12:48:32 -04:00
launch_solr "$FG" "$ADDITIONAL_CMD_OPTS"
2014-08-06 12:30:01 -04:00
else
#
# SolrCloud example is a bit involved so needs special handling here
#
SOLR_SERVER_DIR=$SOLR_TIP/node1
SOLR_HOME=$SOLR_TIP/node1/solr
SOLR_PORT=${CLOUD_PORTS[0]}
2014-10-09 14:42:21 -04:00
2014-09-22 13:23:14 -04:00
if [ "$ZK_HOST" != "" ]; then
DASHZ="-z $ZK_HOST"
fi
2014-10-09 14:42:21 -04:00
if [ "$SOLR_HEAP" != "" ]; then
DASHM="-m $SOLR_HEAP"
fi
if [ "$ADDITIONAL_CMD_OPTS" != "" ]; then
DASHA="-a $ADDITIONAL_CMD_OPTS"
fi
2014-08-06 12:30:01 -04:00
echo -e "\nStarting up SolrCloud node1 on port ${CLOUD_PORTS[0]} using command:\n"
2014-10-09 14:42:21 -04:00
echo -e "solr start -cloud -d node1 -p $SOLR_PORT $DASHZ $DASHM $DASHA\n\n"
2014-08-19 22:56:18 -04:00
2014-08-06 12:30:01 -04:00
# can't launch this node in the foreground else we can't run anymore commands
2014-10-09 14:42:21 -04:00
launch_solr "false" "$ADDITIONAL_CMD_OPTS"
sleep 5
2014-09-22 13:23:14 -04:00
# if user did not define a specific -z parameter, assume embedded in first cloud node we launched above
zk_host=$ZK_HOST
if [ "$zk_host" == "" ]; then
zk_port=$[$SOLR_PORT+1000]
zk_host=localhost:$zk_port
fi
2014-10-09 14:42:21 -04:00
2014-08-06 12:30:01 -04:00
for (( s=1; s<$CLOUD_NUM_NODES; s++ ))
do
ndx=$[$s+1]
next_port=${CLOUD_PORTS[$s]}
echo -e "\n\nStarting node$ndx on port $next_port using command:\n"
2014-10-09 14:42:21 -04:00
echo -e "solr start -cloud -d node$ndx -p $next_port -z $zk_host $DASHM $DASHA \n\n"
2014-08-06 12:30:01 -04:00
# call this script again with correct args for next node
2014-10-09 14:42:21 -04:00
$SOLR_TIP/bin/solr start -cloud -d node$ndx -p $next_port -z $zk_host $DASHM $DASHA
2014-08-06 12:30:01 -04:00
done
# TODO: better (shorter) name??
CLOUD_COLLECTION=gettingstarted
2014-08-19 22:56:18 -04:00
2014-08-06 12:30:01 -04:00
if $noprompt ; then
CLOUD_NUM_SHARDS=2
2014-08-19 22:56:18 -04:00
CLOUD_REPFACT=2
# if the new default config directory is available, then use it,
# otherwise, use the legacy collection1 example
# TODO: this will need to change when SOLR-3619 is resolved
if [ -d "$SOLR_TIP/server/solr/configsets/schemaless" ]; then
CLOUD_CONFIG_DIR=$SOLR_TIP/server/solr/configsets/schemaless
CLOUD_CONFIG=schemaless
else
CLOUD_CONFIG_DIR=$SOLR_TIP/example/solr/collection1/conf
CLOUD_CONFIG=default
fi
2014-08-06 12:30:01 -04:00
else
echo -e "\nNow let's create a new collection for indexing documents in your $CLOUD_NUM_NODES-node cluster.\n"
read -e -p "Please provide a name for your new collection: [gettingstarted] " USER_INPUT
# trim whitespace out of the user input
CLOUD_COLLECTION=`echo $USER_INPUT | tr -d ' '`
# handle the default selection or empty input
if [ "$CLOUD_COLLECTION" == "" ]; then
CLOUD_COLLECTION=gettingstarted
fi
echo $CLOUD_COLLECTION
2014-08-19 22:56:18 -04:00
USER_INPUT=
2014-08-06 12:30:01 -04:00
read -e -p "How many shards would you like to split $CLOUD_COLLECTION into? [2] " USER_INPUT
# trim whitespace out of the user input
CLOUD_NUM_SHARDS=`echo $USER_INPUT | tr -d ' '`
# handle the default selection or empty input
if [ "$CLOUD_NUM_SHARDS" == "" ]; then
CLOUD_NUM_SHARDS=2
fi
echo $CLOUD_NUM_SHARDS
2014-08-19 22:56:18 -04:00
USER_INPUT=
2014-08-06 12:30:01 -04:00
read -e -p "How many replicas per shard would you like to create? [2] " USER_INPUT
# trim whitespace out of the user input
CLOUD_REPFACT=`echo $USER_INPUT | tr -d ' '`
# handle the default selection or empty input
if [ "$CLOUD_REPFACT" == "" ]; then
CLOUD_REPFACT=2
fi
echo $CLOUD_REPFACT
2014-08-19 22:56:18 -04:00
USER_INPUT=
read -e -p "Please choose a configuration for the $CLOUD_COLLECTION collection, available options are: default or schemaless [default] " USER_INPUT
# trim whitespace out of the user input
CLOUD_CONFIG=`echo $USER_INPUT | tr -d ' '`
# handle the default selection or empty input
if [ "$CLOUD_CONFIG" == "" ]; then
CLOUD_CONFIG=default
fi
echo $CLOUD_CONFIG
if [ "$CLOUD_CONFIG" == "schemaless" ]; then
if [ -d "$SOLR_TIP/server/solr/configsets/schemaless" ]; then
CLOUD_CONFIG_DIR=$SOLR_TIP/server/solr/configsets/schemaless
else
CLOUD_CONFIG_DIR=$SOLR_TIP/example/example-schemaless/solr/collection1/conf
fi
else
CLOUD_CONFIG_DIR=$SOLR_TIP/example/solr/collection1/conf
fi
2014-08-06 12:30:01 -04:00
fi
2014-08-19 22:56:18 -04:00
2014-08-06 12:30:01 -04:00
echo -e "\nDeploying default Solr configuration files to embedded ZooKeeper using command:\n"
echo -e "$DEFAULT_SERVER_DIR/scripts/cloud-scripts/zkcli.sh -zkhost $zk_host -cmd upconfig -confdir $CLOUD_CONFIG_DIR -confname $CLOUD_CONFIG\n"
# upload the config directory to ZooKeeper
# Extract the solr.war if it hasn't been done already (so we can access the SolrCLI class)
if [ ! -d "$DEFAULT_SERVER_DIR/solr-webapp/webapp" ]; then
(mkdir -p $DEFAULT_SERVER_DIR/solr-webapp/webapp && cd $DEFAULT_SERVER_DIR/solr-webapp/webapp && jar xf $DEFAULT_SERVER_DIR/webapps/solr.war)
fi
$JAVA -Dlog4j.configuration=file:$DEFAULT_SERVER_DIR/scripts/cloud-scripts/log4j.properties \
-classpath "$DEFAULT_SERVER_DIR/solr-webapp/webapp/WEB-INF/lib/*:$DEFAULT_SERVER_DIR/lib/ext/*" \
org.apache.solr.cloud.ZkCLI -zkhost $zk_host -cmd upconfig -confdir $CLOUD_CONFIG_DIR -confname $CLOUD_CONFIG > /dev/null 2>&1
2014-08-19 22:56:18 -04:00
echo -e "Successfully deployed the $CLOUD_CONFIG_DIR configuration directory to ZooKeeper as $CLOUD_CONFIG\n"
2014-08-06 12:30:01 -04:00
# note use of ceiling logic in case of remainder
MAX_SHARDS_PER_NODE=$((($CLOUD_NUM_SHARDS*$CLOUD_REPFACT+$CLOUD_NUM_NODES-1)/$CLOUD_NUM_NODES))
COLLECTIONS_API=http://localhost:$SOLR_PORT/solr/admin/collections
CLOUD_CREATE_COLLECTION_CMD="$COLLECTIONS_API?action=CREATE&name=$CLOUD_COLLECTION&replicationFactor=$CLOUD_REPFACT&numShards=$CLOUD_NUM_SHARDS&collection.configName=$CLOUD_CONFIG&maxShardsPerNode=$MAX_SHARDS_PER_NODE&wt=json&indent=2"
2014-08-19 22:56:18 -04:00
echo -e "\n\nCreating new collection $CLOUD_COLLECTION with $CLOUD_NUM_SHARDS shards and replication factor $CLOUD_REPFACT using Collections API command:\n\n$CLOUD_CREATE_COLLECTION_CMD\n\nFor more information about the Collections API, please see: https://cwiki.apache.org/confluence/display/solr/Collections+API\n"
curl "$CLOUD_CREATE_COLLECTION_CMD"
2014-08-06 12:30:01 -04:00
echo -e "\n\nSolrCloud example running, please visit http://localhost:$SOLR_PORT/solr \n\n"
fi
exit $?