| #!/bin/bash |
| # |
| # Copyright (c) 2012 The Chromium OS Authors. All rights reserved. |
| # Use of this source code is governed by a BSD-style license that can be |
| # found in the LICENSE file. |
| set -e |
| |
| USAGE='Usage: setup_dev_autotest.sh [-pavnm]' |
| HELP="${USAGE}\n\n\ |
| Install and configure software needed to run autotest locally.\n\ |
| If you're just working on tests, you do not need to run this.\n\n\ |
| Options:\n\ |
| -p Desired Autotest DB password. Must be non-empty.\n\ |
| -a Absolute path to autotest source tree.\n\ |
| -v Show info logging from build_externals.py and compile_gwt_clients.py \n\ |
| -n Non-interactive mode, doesn't ask for any user input. |
| Requires -p and -a to be set.\n\ |
| -m Allow remote access for database." |
| |
| function get_y_or_n_interactive { |
| local ret |
| while true; do |
| read -p "$2" yn |
| case $yn in |
| [Yy]* ) ret="y"; break;; |
| [Nn]* ) ret="n"; break;; |
| * ) echo "Please enter y or n.";; |
| esac |
| done |
| eval $1="'$ret'" |
| } |
| |
| function get_y_or_n { |
| local ret=$3 |
| if [ "${noninteractive}" = "FALSE" ]; then |
| get_y_or_n_interactive sub "$2" |
| ret=$sub |
| fi |
| eval $1="'$ret'" |
| } |
| |
| AUTOTEST_DIR= |
| PASSWD= |
| verbose="FALSE" |
| noninteractive="FALSE" |
| remotedb="FALSE" |
| while getopts ":p:a:vnmh" opt; do |
| case ${opt} in |
| a) |
| AUTOTEST_DIR=$OPTARG |
| ;; |
| p) |
| PASSWD=$OPTARG |
| ;; |
| v) |
| verbose="TRUE" |
| ;; |
| n) |
| noninteractive="TRUE" |
| ;; |
| m) |
| remotedb="TRUE" |
| ;; |
| h) |
| echo -e "${HELP}" >&2 |
| exit 0 |
| ;; |
| \?) |
| echo "Invalid option: -$OPTARG" >&2 |
| echo -e "${HELP}" >&2 |
| exit 1 |
| ;; |
| :) |
| echo "Option -$OPTARG requires an argument." >&2 |
| echo -e "${HELP}" >&2 |
| exit 1 |
| ;; |
| esac |
| done |
| |
| if [[ $EUID -eq 0 ]]; then |
| echo "Running with sudo / as root is not recommended" |
| get_y_or_n verify "Continue as root? [y/N]: " "n" |
| if [[ "${verify}" = 'n' ]]; then |
| echo "Bailing!" |
| exit 1 |
| fi |
| fi |
| |
| if [ "${noninteractive}" = "TRUE" ]; then |
| if [ -z "${AUTOTEST_DIR}" ]; then |
| echo "-a must be specified in non-interactive mode." >&2 |
| exit 1 |
| fi |
| if [ -z "${PASSWD}" ]; then |
| echo "-p must be specified in non-interactive mode." >&2 |
| exit 1 |
| fi |
| fi |
| |
| |
| if [ -z "${PASSWD}" ]; then |
| read -s -p "Autotest DB password: " PASSWD |
| echo |
| if [ -z "${PASSWD}" ]; then |
| echo "Empty passwords not allowed." >&2 |
| exit 1 |
| fi |
| read -s -p "Re-enter password: " PASSWD2 |
| echo |
| if [ "${PASSWD}" != "${PASSWD2}" ]; then |
| echo "Passwords don't match." >&2 |
| exit 1 |
| fi |
| fi |
| |
| if [ -z "${AUTOTEST_DIR}" ]; then |
| CANDIDATE=$(dirname "$(readlink -f "$0")" | egrep -o '(/[^/]+)*/files') |
| read -p "Enter autotest dir [${CANDIDATE}]: " AUTOTEST_DIR |
| if [ -z "${AUTOTEST_DIR}" ]; then |
| AUTOTEST_DIR="${CANDIDATE}" |
| fi |
| fi |
| |
| |
| # Sanity check AUTOTEST_DIR. If it's null, or doesn't exist on the filesystem |
| # then die. |
| if [ -z "${AUTOTEST_DIR}" ]; then |
| echo "No AUTOTEST_DIR. Aborting script." |
| exit 1 |
| fi |
| |
| if [ ! -d "${AUTOTEST_DIR}" ]; then |
| echo "Directory " ${AUTOTEST_DIR} " does not exist. Aborting script." |
| exit 1 |
| fi |
| |
| |
| SHADOW_CONFIG_PATH="${AUTOTEST_DIR}/shadow_config.ini" |
| echo "Autotest supports local overrides of global configuration through a " |
| echo "'shadow' configuration file. Setting one up for you now." |
| CLOBBER=0 |
| if [ -f ${SHADOW_CONFIG_PATH} ]; then |
| get_y_or_n clobber "Clobber existing shadow config? [Y/n]: " "n" |
| if [[ "${clobber}" = 'n' ]]; then |
| CLOBBER=1 |
| echo "Refusing to clobber existing shadow_config.ini." |
| else |
| echo "Clobbering existing shadow_config.ini." |
| fi |
| fi |
| |
| CROS_CHECKOUT=$(readlink -f ${AUTOTEST_DIR}/../../../..) |
| |
| # Create clean shadow config if we're replacing it/creating a new one. |
| if [ $CLOBBER -eq 0 ]; then |
| cat > "${SHADOW_CONFIG_PATH}" <<EOF |
| [AUTOTEST_WEB] |
| host: localhost |
| password: ${PASSWD} |
| readonly_host: localhost |
| readonly_user: chromeosqa-admin |
| readonly_password: ${PASSWD} |
| |
| [SERVER] |
| hostname: localhost |
| |
| [SCHEDULER] |
| drones: localhost |
| |
| [CROS] |
| source_tree: ${CROS_CHECKOUT} |
| EOF |
| echo -e "Done!\n" |
| fi |
| |
| echo "Installing needed Ubuntu packages..." |
| PKG_LIST="mysql-server mysql-common libapache2-mod-wsgi python-mysqldb \ |
| gnuplot apache2-mpm-prefork unzip python-imaging libpng12-dev libfreetype6-dev \ |
| sqlite3 python-pysqlite2 git-core pbzip2 openjdk-6-jre openjdk-6-jdk \ |
| python-crypto python-dev subversion build-essential python-setuptools \ |
| python-numpy python-scipy" |
| |
| if ! sudo apt-get install -y ${PKG_LIST}; then |
| echo "Could not install packages: $?" |
| exit 1 |
| fi |
| echo -e "Done!\n" |
| |
| # Check if database exists, clobber existing database with user consent. |
| # |
| # Arguments: Name of the database |
| check_database() |
| { |
| local db_name=$1 |
| echo "Setting up Database: $db_name in MySQL..." |
| if mysql -u root -e ';' 2> /dev/null ; then |
| PASSWD_STRING= |
| elif mysql -u root -p"${PASSWD}" -e ';' 2> /dev/null ; then |
| PASSWD_STRING="-p${PASSWD}" |
| else |
| PASSWD_STRING="-p" |
| fi |
| |
| if ! mysqladmin -u root "${PASSWD_STRING}" ping ; then |
| sudo service mysql start |
| fi |
| |
| local clobberdb='y' |
| local existing_database=$(mysql -u root "${PASSWD_STRING}" -e "SELECT \ |
| SCHEMA_NAME FROM INFORMATION_SCHEMA.SCHEMATA WHERE SCHEMA_NAME = '$db_name'") |
| |
| if [ -n "${existing_database}" ]; then |
| get_y_or_n clobberdb "Clobber existing MySQL database? [Y/n]: " "n" |
| fi |
| |
| local sql_priv="GRANT ALL PRIVILEGES ON $db_name.* TO \ |
| 'chromeosqa-admin'@'localhost' IDENTIFIED BY '${PASSWD}';" |
| |
| if [ "${remotedb}" = "TRUE" ]; then |
| sql_priv="${sql_priv} GRANT ALL PRIVILEGES ON $db_name.* TO \ |
| 'chromeosqa-admin'@'%' IDENTIFIED BY '${PASSWD}';" |
| fi |
| |
| local sql_command="drop database if exists $db_name; \ |
| create database $db_name; \ |
| ${sql_priv} FLUSH PRIVILEGES;" |
| |
| if [[ "${clobberdb}" = 'y' ]]; then |
| mysql -u root "${PASSWD_STRING}" -e "${sql_command}" |
| fi |
| echo -e "Done!\n" |
| } |
| |
| check_database 'chromeos_autotest_db' |
| check_database 'chromeos_lab_servers' |
| |
| AT_DIR=/usr/local/autotest |
| echo -n "Bind-mounting your autotest dir at ${AT_DIR}..." |
| sudo mkdir -p "${AT_DIR}" |
| sudo mount --bind "${AUTOTEST_DIR}" "${AT_DIR}" |
| echo -e "Done!\n" |
| |
| sudo chown -R "$(whoami)" "${AT_DIR}" |
| |
| EXISTING_MOUNT=$(egrep "/.+[[:space:]]${AT_DIR}" /etc/fstab || /bin/true) |
| if [ -n "${EXISTING_MOUNT}" ]; then |
| echo "${EXISTING_MOUNT}" | awk '{print $1 " already automounting at " $2}' |
| echo "We won't update /etc/fstab, but you should have a line line this:" |
| echo -e "${AUTOTEST_DIR}\t${AT_DIR}\tbind defaults,bind\t0\t0" |
| else |
| echo -n "Adding aforementioned bind-mount to /etc/fstab..." |
| # Is there a better way to elevate privs and do a redirect? |
| sudo su -c \ |
| "echo -e '${AUTOTEST_DIR}\t${AT_DIR}\tbind defaults,bind\t0\t0' \ |
| >> /etc/fstab" |
| echo -e "Done!\n" |
| fi |
| |
| echo -n "Reticulating splines..." |
| |
| if [ "${verbose}" = "TRUE" ]; then |
| "${AT_DIR}"/utils/build_externals.py |
| "${AT_DIR}"/utils/compile_gwt_clients.py -a |
| else |
| "${AT_DIR}"/utils/build_externals.py &> /dev/null |
| "${AT_DIR}"/utils/compile_gwt_clients.py -a &> /dev/null |
| fi |
| |
| echo -e "Done!\n" |
| |
| echo "Populating autotest mysql DB..." |
| "${AT_DIR}"/database/migrate.py sync -f |
| "${AT_DIR}"/frontend/manage.py syncdb --noinput |
| # You may have to run this twice. |
| "${AT_DIR}"/frontend/manage.py syncdb --noinput |
| "${AT_DIR}"/utils/test_importer.py |
| echo -e "Done!\n" |
| |
| echo "Initializing chromeos_lab_servers mysql DB..." |
| "${AT_DIR}"/database/migrate.py sync -f -d AUTOTEST_SERVER_DB |
| echo -e "Done!\n" |
| |
| echo "Configuring apache to run the autotest web interface..." |
| if [ ! -d /etc/apache2/run ]; then |
| sudo mkdir /etc/apache2/run |
| fi |
| sudo ln -sf "${AT_DIR}"/apache/apache-conf \ |
| /etc/apache2/sites-available/autotest-server.conf |
| # Disable currently active default |
| sudo a2dissite 000-default default || true |
| # Enable autotest server |
| sudo a2ensite autotest-server.conf |
| # Enable rewrite module |
| sudo a2enmod rewrite |
| # Enable wsgi |
| sudo a2enmod wsgi |
| # Enable version |
| # built-in on trusty |
| sudo a2enmod version || true |
| # Enable headers |
| sudo a2enmod headers |
| # Enable cgid |
| sudo a2enmod cgid |
| # Setup permissions so that Apache web user can read the proper files. |
| chmod -R o+r "${AT_DIR}" |
| find "${AT_DIR}"/ -type d -print0 | xargs --null chmod o+x |
| chmod o+x "${AT_DIR}"/tko/*.cgi |
| # restart server |
| sudo /etc/init.d/apache2 restart |
| |
| # Setup lxc and base container for server-side packaging support. |
| sudo apt-get install lxc -y |
| sudo python "${AT_DIR}"/site_utils/lxc.py -s |
| |
| echo "Browse to http://localhost to see if Autotest is working." |
| echo "For further necessary set up steps, see https://sites.google.com/a/chromium.org/dev/chromium-os/testing/autotest-developer-faq/setup-autotest-server?pli=1" |