blob: 94e11ec788d20ef26d7e975e89fad242d7039782 [file] [log] [blame]
#!/bin/bash
#
# Copyright (C) 2015 The Android Open Source Project
#
# Licensed 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.
#
# Version: 1.3-a5
#
set -o nounset
BASE_UMASK=$(umask)
umask 077
#
# Settings
#
JACK_HOME="${JACK_HOME:=$HOME/.jack-server}"
CLIENT_SETTING="${CLIENT_SETTING:=$HOME/.jack-settings}"
TMPDIR=${TMPDIR:=/tmp}
JACK_SERVER_VM_ARGUMENTS="${JACK_SERVER_VM_ARGUMENTS:=-Dfile.encoding=UTF-8 -XX:+TieredCompilation}"
LAUNCHER_JAR="$JACK_HOME/launcher.jar"
LAUNCHER_NAME=com.android.jack.launcher.ServerLauncher
CURRENT_CHARSET=$(locale charmap)
JACK_LOGS_DIR="$JACK_HOME"/logs
JACK_OUT_ERR="$JACK_LOGS_DIR"/outputs.txt
JACK_CONNECTION_TIMEOUT=300
#
# Load client settings
#
if [ -f "$CLIENT_SETTING" ]; then
source "$CLIENT_SETTING"
fi
#
# Create or update client settings if needed
#
if [[ ! -f "$CLIENT_SETTING" || $SETTING_VERSION -lt 4 ]]; then
echo "Writing client settings in" $CLIENT_SETTING
cat >"$CLIENT_SETTING.$$" <<-EOT
# Server settings
SERVER_HOST=${SERVER_HOST:=127.0.0.1}
SERVER_PORT_SERVICE=${SERVER_PORT_SERVICE:=8076}
SERVER_PORT_ADMIN=${SERVER_PORT_ADMIN:=8077}
# Internal, do not touch
SETTING_VERSION=4
EOT
ln -f "$CLIENT_SETTING.$$" "$CLIENT_SETTING"
rm "$CLIENT_SETTING.$$"
source "$CLIENT_SETTING"
fi
usage () {
echo "Usage : $0 [ install-server <launcher.jar> <server.jar> | uninstall-server | list <program> | update <program> <program.jar> | start-server | stop-server | kill-server | list-server | server-stat | server-log | server-gc | cleanup-server]"
}
abort () { exit 255; }
#
# $1: curl command status
# $2: HTTP status
#
handleHttpErrors() {
if [ $1 -eq 0 ]; then
# No problem, let's go
return 0;
elif [ $1 -eq 7 ]; then
echo "No Jack server running. Try 'jack-admin start-server'" >&2
abort
elif [ $1 -eq 35 ]; then
echo "SSL error when connecting to the Jack server" >&2
abort
elif [ $1 -eq 58 ]; then
echo "Failed to contact Jack server: Problem reading ${JACK_HOME}/client.pem" >&2
abort
elif [ $1 -eq 60 ]; then
echo "Failed to authenticate Jack server certificate" >&2
abort
elif [ $1 -eq 77 ]; then
echo "Failed to contact Jack server: Problem reading ${JACK_HOME}/server.pem" >&2
abort
elif [ $1 -eq 22 ]; then
# Http code not OK, let's decode and abort
if [ $2 -eq 400 ]; then
# 400: Bad request
echo "Bad request, see Jack server log" >&2
abort
else
# Other
echo "Internal unknown error ($2), try other ports or see Jack server log" >&2
abort
fi
else
echo "Communication error with Jack server $1" >&2
abort
fi
}
checkCurlVersion () {
curl --version | grep -q "SecureTransport"
if [ "$?" -eq 0 ]; then
echo "Unsupported curl, please use a curl not based on SecureTransport" >&2
abort
fi
}
#
# $1: program name
# $2: jar of the program
#
updateProgram () {
HTTP_CODE=$(curl -f \
--cert "${JACK_HOME}/client.pem" \
--cacert "${JACK_HOME}/server.pem" \
--output >(cat >/dev/null) \
--no-buffer --write-out '%{http_code}' --silent --connect-timeout $JACK_CONNECTION_TIMEOUT \
-X PUT \
-F "jar=@$2;type=application/octet-stream" \
-F "force=$FORCE_INSTALLATION;type=text/plain;charset=$CURRENT_CHARSET" \
--noproxy ${SERVER_HOST} \
https://${SERVER_HOST}:$SERVER_PORT_ADMIN/$1 \
)
handleHttpErrors $? $HTTP_CODE
if [ "$1" == server ]; then
echo "Server updated, waiting for restart"
waitServerStarted
fi
}
isServerRunning () {
RETRY_SESSION=3
DONE=1
let DATE_TIMEOUT=$(date +%s)+$JACK_CONNECTION_TIMEOUT
while [ "$DONE" -ne 0 ]; do
HTTP_CODE=$(curl -f \
--cert "${JACK_HOME}/client.pem" \
--cacert "${JACK_HOME}/server.pem" \
--output >(cat >/dev/null) \
--no-buffer --write-out '%{http_code}' --silent --connect-timeout $JACK_CONNECTION_TIMEOUT \
-X GET \
-H "Accept: text/plain;charset=$CURRENT_CHARSET" \
--noproxy ${SERVER_HOST} \
https://${SERVER_HOST}:$SERVER_PORT_ADMIN/server \
)
CURL_CODE=$?
if [ $CURL_CODE -eq 0 ]; then
# No problem, let's go
return 0;
elif [ $CURL_CODE -eq 7 ]; then
return 1
else
# In case of partial, timeout, empty response, network error, let's retry
if [ $RETRY_SESSION -eq 0 ]; then
echo "Communication error with Jack server ($CURL_CODE), try other ports or see Jack server log" >&2
abort
else
if [ $(date +%s) -lt $DATE_TIMEOUT ]; then
let RETRY_SESSION=RETRY_SESSION-1
else
echo "Communication error with Jack server ($CURL_CODE), try other ports or see Jack server log" >&2
abort
fi
fi
fi
done
}
waitServerStarted () {
DONE=1
let DATE_TIMEOUT=$(date +%s)+$JACK_CONNECTION_TIMEOUT
while [ "$DONE" -ne 0 ]; do
HTTP_CODE=$(curl -f \
--cert "${JACK_HOME}/client.pem" \
--cacert "${JACK_HOME}/server.pem" \
--output >(cat >/dev/null) \
--no-buffer --write-out '%{http_code}' --silent --connect-timeout $JACK_CONNECTION_TIMEOUT \
-X GET \
-H "Accept: text/plain;charset=$CURRENT_CHARSET" \
--noproxy ${SERVER_HOST} \
https://${SERVER_HOST}:$SERVER_PORT_ADMIN/server \
)
CURL_CODE=$?
if [ $CURL_CODE -eq 7 ] || [ $CURL_CODE -eq 35 ] || [ $CURL_CODE -eq 58 ] || [ $CURL_CODE -eq 60 ] || [ $CURL_CODE -eq 77 ]; then
if [ $(date +%s) -ge $DATE_TIMEOUT ]; then
echo "Jack server failed to (re)start, see Jack server log" >&2
abort
else
sleep 1
fi
else
# A connection was opened, no need to know if it went well
DONE=0;
fi
done
}
#
# $1: program name
#
listProgramVersion () {
exec 3>&1
HTTP_CODE=$(curl -f \
--cert "${JACK_HOME}/client.pem" \
--cacert "${JACK_HOME}/server.pem" \
--output >(tr -d '\015' >&3) \
--no-buffer --write-out '%{http_code}' --silent --connect-timeout $JACK_CONNECTION_TIMEOUT \
-X GET \
-H "Accept: text/plain;charset=$CURRENT_CHARSET" \
--noproxy ${SERVER_HOST} \
https://${SERVER_HOST}:$SERVER_PORT_ADMIN/$1 \
)
handleHttpErrors $? $HTTP_CODE
exec 3>&-
}
#
# Decoding argument
#
if [ $# -eq 0 ]
then
usage
abort
fi
set +o errexit
FORCE_INSTALLATION=false
case $1 in
force-update)
FORCE_INSTALLATION=true
COMMAND=update;;
*)
COMMAND=$1;;
esac
case $COMMAND in
install-server)
if [ $# -ne 3 ]; then
usage
abort
fi
if [ ! -r "$2" ]; then
echo "Jack server launcher jar \"$2\" is not readable" >&2
abort
fi
if [ ! -r "$3" ]; then
echo "Jack server jar \"$3\" is not readable" >&2
abort
fi
checkCurlVersion
if [ ! -d "$JACK_HOME" ]; then
echo Installing jack server in \"$JACK_HOME\"
mkdir -p "$JACK_HOME"
cp $2 "$LAUNCHER_JAR"
cp $3 "$JACK_HOME/server-1.jar"
mkdir "$JACK_LOGS_DIR"
keytool -genkeypair -validity 3650 -alias server -keyalg RSA -keysize 2048 -keypass Jack-Server -storepass Jack-Server -dname "CN=$SERVER_HOST" -keystore "$JACK_HOME/server.jks"
keytool -genkeypair -validity 3650 -alias client -keyalg RSA -keysize 2048 -keypass Jack-Server -storepass Jack-Server -dname "CN=$(id -un)@$(uname -n)" -keystore "$JACK_HOME/client.jks"
else
echo "Jack server already installed in \"$JACK_HOME\"" >&2
abort
fi
exit 0 ;;
uninstall-server)
if [ ! -d "$JACK_HOME" ]; then
echo "Jack server in \"$JACK_HOME\" not found" >&2
abort
else
echo "Removing jack server from \"$JACK_HOME\""
rm -rf "$JACK_HOME"
fi
exit 0 ;;
list)
if [ $# -ne 2 ]
then
usage
abort
fi
listProgramVersion $2 ;;
update)
if [ $# -lt 3 ]; then
usage
abort
fi
if [ $# -gt 4 ]; then
usage
abort
fi
if [ ! -r "$3" ]; then
echo "Failed to update $2 of Jack server: \"$3\" is not readable" >&2
abort
fi
checkCurlVersion
if [ $FORCE_INSTALLATION = true ]; then
updateProgram $2 $3
else
if [ $# -eq 4 ]; then
HTTP_CODE=$(curl -f \
--cert "${JACK_HOME}/client.pem" \
--cacert "${JACK_HOME}/server.pem" \
--output >(cat >/dev/null) \
--no-buffer --write-out '%{http_code}' --silent --connect-timeout $JACK_CONNECTION_TIMEOUT \
-X HEAD \
--data "$4" \
-H "Content-Type:application/vnd.jack.select-exact;version=1" \
--noproxy ${SERVER_HOST} \
https://${SERVER_HOST}:$SERVER_PORT_ADMIN/$2 \
)
CURL_CODE=$?
if [ $CURL_CODE -eq 22 ]; then
if [ $HTTP_CODE -eq 404 ]; then
# version not found, proceed to installation
updateProgram $2 $3
exit 0
fi
fi
handleHttpErrors $CURL_CODE $HTTP_CODE
else
# No version provided, proceed directly without testing
updateProgram $2 $3
fi
fi
exit 0;;
stop-server)
echo "Stopping background server"
HTTP_CODE=$(curl -f \
--cert "${JACK_HOME}/client.pem" \
--cacert "${JACK_HOME}/server.pem" \
--output >(cat >/dev/null) \
--no-buffer --write-out '%{http_code}' --silent --connect-timeout $JACK_CONNECTION_TIMEOUT \
-X POST \
--noproxy ${SERVER_HOST} \
https://${SERVER_HOST}:$SERVER_PORT_ADMIN/server/stop \
)
handleHttpErrors $? $HTTP_CODE ;;
server-stat)
echo "Getting statistic from background server"
exec 3>&1
HTTP_CODE=$(curl -f \
--cert "${JACK_HOME}/client.pem" \
--cacert "${JACK_HOME}/server.pem" \
--output >(tr -d '\015' >&3) \
--no-buffer --write-out '%{http_code}' --silent --connect-timeout $JACK_CONNECTION_TIMEOUT \
-X GET \
-H "Accept: text/plain;charset=$CURRENT_CHARSET" \
--noproxy ${SERVER_HOST} \
https://${SERVER_HOST}:$SERVER_PORT_ADMIN/stat \
)
handleHttpErrors $? $HTTP_CODE
exec 3>&- ;;
server-log)
exec 3>&1
HTTP_CODE=$(curl -f \
--cert "${JACK_HOME}/client.pem" \
--cacert "${JACK_HOME}/server.pem" \
--output >(tr -d '\015' >&3) \
--no-buffer --write-out '%{http_code}' --silent --connect-timeout $JACK_CONNECTION_TIMEOUT \
-X GET \
-H "Accept: text/plain;charset=$CURRENT_CHARSET" \
--noproxy ${SERVER_HOST} \
https://${SERVER_HOST}:$SERVER_PORT_ADMIN/launcher/log \
)
handleHttpErrors $? $HTTP_CODE
exec 3>&- ;;
kill-server)
echo "Killing background server"
kill $(ps aux | grep $LAUNCHER_NAME | grep -v grep | awk '{print $2}') 2>/dev/null
if [ $? -ne 0 ]; then
echo "No Jack server to kill" >&2
exit 2
else
exit 0
fi ;;
list-server)
ps aux | grep $LAUNCHER_NAME | grep -v grep
exit $? ;;
start-server)
isServerRunning
RUNNING=$?
if [ "$RUNNING" = 0 ]; then
echo "Server is already running"
else
JACK_SERVER_COMMAND="java -Djava.io.tmpdir=$TMPDIR $JACK_SERVER_VM_ARGUMENTS -cp $LAUNCHER_JAR $LAUNCHER_NAME"
echo "Launching Jack server" $JACK_SERVER_COMMAND
(
trap "" SIGHUP
for i in $(seq 3 255); do
eval exec "$i"'>&-'
done
cd "$JACK_HOME"
umask $BASE_UMASK
exec $JACK_SERVER_COMMAND
abort
) >"$JACK_OUT_ERR" 2>&1 &
fi
waitServerStarted
exit 0 ;;
server-log-level)
if [ $# -eq 4 ]
then
LIMIT=$3
COUNT=$4
elif [ $# -eq 2 ]
then
COUNT=2
if [ \( "$2" = "ERROR" \) -o \( "$2" = "WARNING" \) ]
then
LIMIT=1048576
else
LIMIT=10485760
fi
else
usage
abort
fi
echo "Setting logging parameters of background server"
HTTP_CODE=$(curl --fail \
--cert "${JACK_HOME}/client.pem" \
--cacert "${JACK_HOME}/server.pem" \
--output >(cat >/dev/null) \
--no-buffer --write-out '%{http_code}' --silent --connect-timeout $JACK_CONNECTION_TIMEOUT \
--request PUT \
--form "level=$2;type=text/plain;charset=$CURRENT_CHARSET" \
--form "limit=$LIMIT;type=text/plain;charset=$CURRENT_CHARSET" \
--form "count=$COUNT;type=text/plain;charset=$CURRENT_CHARSET" \
--noproxy ${SERVER_HOST} \
https://${SERVER_HOST}:$SERVER_PORT_ADMIN/launcher/log/level \
)
handleHttpErrors $? $HTTP_CODE ;;
server-gc)
echo "Requesting a garbage collection to the background server"
HTTP_CODE=$(curl -f \
--cert "${JACK_HOME}/client.pem" \
--cacert "${JACK_HOME}/server.pem" \
--output >(cat >/dev/null) \
--no-buffer --write-out '%{http_code}' --silent --connect-timeout $JACK_CONNECTION_TIMEOUT \
-X POST \
--noproxy ${SERVER_HOST} \
https://${SERVER_HOST}:$SERVER_PORT_ADMIN/gc \
)
handleHttpErrors $? $HTTP_CODE ;;
#
# Should be run when server is off. Allows to clean files that could be forgotten on disk in case of
# server VM crash after an update.
#
cleanup-server)
shopt -s nullglob
for file in $JACK_HOME/jack/*.deleted; do
rm "${file%.deleted}"
rm "$file"
done
exit 0 ;;
*)
usage
abort ;;
esac
# Exit
exit 0