| #!/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 |