#!/bin/sh ## @file # VirtualBox Validation Kit - TestBoxScript service init script. # # # Copyright (C) 2006-2014 Oracle Corporation # # This file is part of VirtualBox Open Source Edition (OSE), as # available from http://www.virtualbox.org. This file is free software; # you can redistribute it and/or modify it under the terms of the GNU # General Public License (GPL) as published by the Free Software # Foundation, in version 2 as it comes in the "COPYING" file of the # VirtualBox OSE distribution. VirtualBox OSE is distributed in the # hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. # # The contents of this file may alternatively be used under the terms # of the Common Development and Distribution License Version 1.0 # (CDDL) only, as it comes in the "COPYING.CDDL" file of the # VirtualBox OSE distribution, in which case the provisions of the # CDDL are applicable instead of those of the GPL. # # You may elect to license modified versions of this file under the # terms and conditions of either the GPL or the CDDL or both. # # chkconfig: 35 35 65 # description: TestBoxScript service # ### BEGIN INIT INFO # Provides: testboxscript-service # Required-Start: $network # Required-Stop: # Default-Start: 2 3 4 5 # Default-Stop: 0 1 6 # Description: TestBoxScript service ### END INIT INFO PATH=$PATH:/bin:/sbin:/usr/sbin # # Load config and set up defaults. # service_name="testboxscript" [ -r /etc/default/${service_name} ] && . /etc/default/${service_name} [ -r /etc/conf.d/${service_name} ] && . /etc/conf.d/${service_name} if [ -z "${TESTBOXSCRIPT_DIR}" ]; then TESTBOXSCRIPT_DIR="/opt/testboxscript" fi if [ -z "${TESTBOXSCRIPT_USER}" ]; then TESTBOXSCRIPT_USER="vbox" fi binary="${TESTBOXSCRIPT_DIR}/testboxscript/testboxscript.py" binary_real="${TESTBOXSCRIPT_DIR}/testboxscript/testboxscript_real.py" # # Detect and abstract distro # [ -f /etc/debian_release -a -f /lib/lsb/init-functions ] || NOLSB=yes system=unknown if [ -f /etc/redhat-release ]; then system=redhat PIDFILE="/var/run/${service_name}-service.pid" elif [ -f /etc/SuSE-release ]; then system=suse PIDFILE="/var/lock/subsys/${service_name}-service" elif [ -f /etc/debian_version ]; then system=debian PIDFILE="/var/run/${service_name}-service" elif [ -f /etc/gentoo-release ]; then system=gentoo PIDFILE="/var/run/${service_name}-service" elif [ -f /etc/arch-release ]; then system=arch PIDFILE="/var/run/${service_name}-service" elif [ -f /etc/slackware-version ]; then system=slackware PIDFILE="/var/run/${service_name}-service" elif [ -f /etc/lfs-release ]; then system=lfs PIDFILE="/var/run/${service_name}-service.pid" else system=other if [ -d /var/run -a -w /var/run ]; then PIDFILE="/var/run/${service_name}-service" fi fi # # Generic implementation. # ## Query daemon status. # $1 = daemon-user; $2 = binary name # returns 0 if running, 1 if started but no longer running, 3 if not started. # When 0 is return the pid variable contains a list of relevant pids. my_query_status() { a_USER="$1"; a_BINARY="$2"; pid=""; if [ -f "${PIDFILE}" -a -s "${PIDFILE}" ]; then MY_LINE=""; read MY_LINE < "${PIDFILE}"; for MY_PID in `echo $MY_LINE | sed -e 's/[^0123456789 ]/ /g'`; do if [ "`stat -c '%U' /proc/$MY_PID 2> /dev/null `" = "$a_USER" ]; then pid="${pid} ${MY_PID}"; fi done if [ -n "${pid}" ]; then RETVAL=0; else RETVAL=1; fi else RETVAL=3 fi return $RETVAL; } ## Starts detached daeamon in screen. # $1 = daemon-user; $2+ = daemon and its arguments my_start_daemon() { a_USER="$1" shift if touch "${PIDFILE}" && chown "${a_USER}" -- "${PIDFILE}"; then ARGS="" while [ $# -gt 0 ]; do ARGS="$ARGS '$1'"; shift done ARGS="$ARGS --pidfile '$PIDFILE'"; su - "${a_USER}" -c "screen -S ${service_name} -d -m ${ARGS} "; RETVAL=$?; if [ $RETVAL -eq 0 ]; then sleep 0.6; if [ ! -s "$PIDFILE" ]; then sleep 1; fi if [ ! -s "$PIDFILE" ]; then sleep 2; fi if [ ! -s "$PIDFILE" ]; then sleep 3; fi if [ -s "$PIDFILE" ]; then RETVAL=0; else RETVAL=1; fi else fail_msg "su failed with exit code $RETVAL"; fi else fail_msg "Failed to create pid file and change it's ownership to ${a_USER}." RETVAL=1; fi return $RETVAL; } ## Stops the daemon. # $1 = daemon-user; $2 = binary name my_stop_daemon() { a_USER="$1"; a_BINARY="$2"; my_query_status "$a_USER" "$a_BINARY" RETVAL=$? if [ $RETVAL -eq 0 -a -n "$pid" ]; then kill $pid; fi sleep 0.6 if my_query_status "$a_USER" "$a_BINARY"; then sleep 1; fi if my_query_status "$a_USER" "$a_BINARY"; then sleep 2; fi if my_query_status "$a_USER" "$a_BINARY"; then sleep 3; fi if ! my_query_status "$a_USER" "$a_BINARY"; then rm -f -- "${PIDFILE}" return 0; fi return 1; } if [ -z "$NOLSB" ]; then . /lib/lsb/init-functions fail_msg() { echo "" log_failure_msg "$1" } succ_msg() { log_success_msg " done." } begin_msg() { log_daemon_msg "$@" } else fail_msg() { echo " ...fail!" echo "$@" } succ_msg() { echo " ...done." } begin_msg() { echo -n "$1" } fi # # System specific overrides. # if [ "$system" = "redhat" ]; then . /etc/init.d/functions if [ -n "$NOLSB" ]; then fail_msg() { echo_failure echo } succ_msg() { echo_success echo } begin_msg() { echo -n "$1" } fi fi if [ "$system" = "suse" ]; then . /etc/rc.status if [ -n "$NOLSB" ]; then fail_msg() { rc_failed 1 rc_status -v } succ_msg() { rc_reset rc_status -v } begin_msg() { echo -n "$1" } fi fi if [ "$system" = "debian" ]; then # Share my_start_daemon and my_stop_daemon with gentoo if [ -n "$NOLSB" ]; then fail_msg() { echo " ...fail!" } succ_msg() { echo " ...done." } begin_msg() { echo -n "$1" } fi fi if [ "$system" = "gentoo" ]; then if [ -f /sbin/functions.sh ]; then . /sbin/functions.sh elif [ -f /etc/init.d/functions.sh ]; then . /etc/init.d/functions.sh fi # Share my_start_daemon and my_stop_daemon with debian. if [ -n "$NOLSB" ]; then if [ "`which $0`" = "/sbin/rc" ]; then shift fi fi fi if [ "$system" = "debian" -o "$system" = "gentoo" ]; then #my_start_daemon() { # usr="$1" # shift # bin="$1" # shift # echo usr=$usr # start-stop-daemon --start --background --pidfile "${PIDFILE}" --make-pidfile --chuid "${usr}" --user "${usr}" \ # --exec $bin -- $@ #} my_stop_daemon() { a_USER="$1" a_BINARY="$2" start-stop-daemon --stop --user "${a_USER}" --pidfile "${PIDFILE}" RETVAL=$? rm -f "${PIDFILE}" return $RETVAL } fi if [ "$system" = "arch" ]; then USECOLOR=yes . /etc/rc.d/functions if [ -n "$NOLSB" ]; then fail_msg() { stat_fail } succ_msg() { stat_done } begin_msg() { stat_busy "$1" } fi fi if [ "$system" = "lfs" ]; then . /etc/rc.d/init.d/functions if [ -n "$NOLSB" ]; then fail_msg() { echo_failure } succ_msg() { echo_ok } begin_msg() { echo $1 } fi fi # # Implement the actions. # check_single_user() { if [ -n "$2" ]; then fail_msg "TESTBOXSCRIPT_USER must not contain multiple users!" exit 1 fi } # # Open ports at the firewall: # 6000..6100 / TCP for VRDP # 5000..5032 / TCP for netperf # 5000..5032 / UDP for netperf # set_iptables() { if [ -x /sbin/iptables ]; then I="/sbin/iptables -j ACCEPT -A INPUT -m state --state NEW" if ! /sbin/iptables -L INPUT | grep -q "testsuite vrdp"; then $I -m tcp -p tcp --dport 6000:6100 -m comment --comment "testsuite vrdp" fi if ! /sbin/iptables -L INPUT | grep -q "testsuite perftcp"; then $I -m tcp -p tcp --dport 5000:5032 -m comment --comment "testsuite perftcp" fi if ! /sbin/iptables -L INPUT | grep -q "testsuite perfudp"; then $I -m udp -p udp --dport 5000:5032 -m comment --comment "testsuite perfudp" fi fi } start() { if [ ! -f "${PIDFILE}" ]; then begin_msg "Starting TestBoxScript"; # # Verify config and installation. # if [ ! -d "$TESTBOXSCRIPT_DIR" -o ! -r "$binary" -o ! -r "$binary_real" ]; then fail_msg "Cannot find TestBoxScript installation under '$TESTBOXSCRIPT_DIR'!" exit 0; fi ## @todo check ownership (for upgrade purposes) check_single_user $TESTBOXSCRIPT_USER # # Open some ports in the firewall # Allows to access VMs remotely by VRDP, netperf # set_iptables # # Set execute bits to make installation (unzip) easier. # chmod a+x > /dev/null 2>&1 \ "${binary}" \ "${binary_real}" \ "${TESTBOXSCRIPT_DIR}/linux/amd64/TestBoxHelper" \ "${TESTBOXSCRIPT_DIR}/linux/x86/TestBoxHelper" # # Start the daemon as the specified user. # PARAMS="" if [ "${TESTBOXSCRIPT_HWVIRT}" = "yes" ]; then PARAMS="${PARAMS} --hwvirt"; fi if [ "${TESTBOXSCRIPT_HWVIRT}" = "no" ]; then PARAMS="${PARAMS} --no-hwvirt"; fi if [ "${TESTBOXSCRIPT_NESTED_PAGING}" = "yes" ]; then PARAMS="${PARAMS} --nested-paging"; fi if [ "${TESTBOXSCRIPT_NESTED_PAGING}" = "no" ]; then PARAMS="${PARAMS} --no-nested-paging"; fi if [ "${TESTBOXSCRIPT_IOMMU}" = "yes" ]; then PARAMS="${PARAMS} --io-mmu"; fi if [ "${TESTBOXSCRIPT_IOMMU}" = "no" ]; then PARAMS="${PARAMS} --no-io-mmu"; fi if [ -n "${TESTBOXSCRIPT_SYSTEM_UUID}" ]; then PARAMS="${PARAMS} --system-uuid '${TESTBOXSCRIPT_SYSTEM_UUID}'"; fi if [ -n "${TESTBOXSCRIPT_TEST_MANAGER}" ]; then PARAMS="${PARAMS} --test-manager '${TESTBOXSCRIPT_TEST_MANAGER}'"; fi if [ -n "${TESTBOXSCRIPT_SCRATCH_ROOT}" ]; then PARAMS="${PARAMS} --scratch-root '${TESTBOXSCRIPT_SCRATCH_ROOT}'"; fi if [ -n "${TESTBOXSCRIPT_BUILDS_PATH}" ]; then PARAMS="${PARAMS} --builds-path '${TESTBOXSCRIPT_BUILDS_PATH}'"; fi if [ -n "${TESTBOXSCRIPT_BUILDS_TYPE}" ]; then PARAMS="${PARAMS} --builds-server-type '${TESTBOXSCRIPT_BUILDS_TYPE}'"; fi if [ -n "${TESTBOXSCRIPT_BUILDS_NAME}" ]; then PARAMS="${PARAMS} --builds-server-name '${TESTBOXSCRIPT_BUILDS_NAME}'"; fi if [ -n "${TESTBOXSCRIPT_BUILDS_SHARE}" ]; then PARAMS="${PARAMS} --builds-server-share '${TESTBOXSCRIPT_BUILDS_SHARE}'"; fi if [ -n "${TESTBOXSCRIPT_BUILDS_USER}" ]; then PARAMS="${PARAMS} --builds-server-user '${TESTBOXSCRIPT_BUILDS_USER}'"; fi if [ -n "${TESTBOXSCRIPT_BUILDS_PASSWD}" ]; then PARAMS="${PARAMS} --builds-server-passwd '${TESTBOXSCRIPT_BUILDS_PASSWD}'"; fi if [ -n "${TESTBOXSCRIPT_TESTRSRC_PATH}" ]; then PARAMS="${PARAMS} --testrsrc-path '${TESTBOXSCRIPT_PATH_TESTRSRC}'"; fi if [ -n "${TESTBOXSCRIPT_BUILDS_TYPE}" ]; then PARAMS="${PARAMS} --testrsrc-server-type '${TESTBOXSCRIPT_TESTRSRC_TYPE}'"; fi if [ -n "${TESTBOXSCRIPT_BUILDS_NAME}" ]; then PARAMS="${PARAMS} --testrsrc-server-name '${TESTBOXSCRIPT_TESTRSRC_NAME}'"; fi if [ -n "${TESTBOXSCRIPT_BUILDS_SHARE}" ]; then PARAMS="${PARAMS} --testrsrc-server-share '${TESTBOXSCRIPT_TESTRSRC_SHARE}'"; fi if [ -n "${TESTBOXSCRIPT_BUILDS_USER}" ]; then PARAMS="${PARAMS} --testrsrc-server-user '${TESTBOXSCRIPT_TESTRSRC_USER}'"; fi if [ -n "${TESTBOXSCRIPT_BUILDS_PASSWD}" ]; then PARAMS="${PARAMS} --testrsrc-server-passwd '${TESTBOXSCRIPT_TESTRSRC_PASSWD}'"; fi if [ -n "${TESTBOXSCRIPT_PYTHON}" ]; then my_start_daemon "${TESTBOXSCRIPT_USER}" "${TESTBOXSCRIPT_PYTHON}" "${binary}" ${PARAMS} else my_start_daemon "${TESTBOXSCRIPT_USER}" "${binary}" ${PARAMS} fi RETVAL=$? if [ $RETVAL -eq 0 ]; then succ_msg else fail_msg fi else succ_msg "Already running." RETVAL=0 fi return $RETVAL } stop() { if [ -f "${PIDFILE}" ]; then begin_msg "Stopping TestBoxScript"; my_stop_daemon "${TESTBOXSCRIPT_USER}" "${binary}" RETVAL=$? if [ $RETVAL -eq 0 ]; then succ_msg else fail_msg fi else RETVAL=0 fi return $RETVAL } restart() { stop && sleep 1 && start } status() { echo -n "Checking for TestBoxScript" my_query_status "${TESTBOXSCRIPT_USER}" "${binary}" RETVAL=$? if [ ${RETVAL} -eq 0 ]; then echo " ...running" elif [ ${RETVAL} -eq 3 ]; then echo " ...stopped" elif [ ${RETVAL} -eq 1 ]; then echo " ...started but not running" else echo " ...unknown status '${RETVAL}'" fi } # # main(). # case "$1" in start) start ;; stop) stop ;; restart) restart ;; force-reload) restart ;; status) status ;; setup) ;; cleanup) ;; *) echo "Usage: $0 {start|stop|restart|status}" exit 1 esac exit $RETVAL