From 5f52d63247cb354c6d15bec16f3b11eb96597491 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Juanjo=20L=C3=B3pez?= Date: Fri, 30 Aug 2024 16:47:13 +0200 Subject: [PATCH] Improved test with service checking and browseability --- .github/workflows/tests.yml | 20 +- commands/host/sync-from-env | 323 -------------------------------- scripts/fix-adminer-norouter.sh | 10 - tests/test.bats | 69 +++++-- 4 files changed, 73 insertions(+), 349 deletions(-) delete mode 100644 commands/host/sync-from-env delete mode 100755 scripts/fix-adminer-norouter.sh diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml index f8c42b3..d731b2c 100644 --- a/.github/workflows/tests.yml +++ b/.github/workflows/tests.yml @@ -20,6 +20,19 @@ permissions: actions: write jobs: + code_quality: + strategy: + matrix: + ddev_version: [stable, HEAD] + fail-fast: false + runs-on: ubuntu-latest + steps: + - name: "Check out code" + uses: actions/checkout@v3 + - name: "ShellChecker" + uses: a5k-actions/shellchecker@v0 + with: + severity: error tests: strategy: matrix: @@ -29,11 +42,8 @@ jobs: runs-on: ubuntu-latest steps: - - name: Run ShellCheck - uses: azohra/shell-linter@latest - with: - path: "scripts,commands" - + - name: Action Setup Bash + uses: release-engineers/action-setup-bash@v1 - uses: ddev/github-action-add-on-test@v2 with: ddev_version: ${{ matrix.ddev_version }} diff --git a/commands/host/sync-from-env b/commands/host/sync-from-env deleted file mode 100644 index 309ac7f..0000000 --- a/commands/host/sync-from-env +++ /dev/null @@ -1,323 +0,0 @@ -#!/bin/bash -#ddev-generated -## Description: Sync local environment with given remote environment. -## Usage: sync-local -## Example: ddev sync-local pro - -####################################### -# Sync local environment with remote environment. -# -# It is recomended to run this script when starting a new branch. -# It should be run outside of docker. -# It currently perform: -# - composer install -# - database sync -# - database sanitization -# - drush updatedb -# _ drush config-import -# - CSS and JS transpilation -# - cache rebuild -# - display a one time user login link -####################################### -set -e - -function get_default_value() { - VARNAME=$1 - DEFAULT_VALUE=$2 - - ENV_VALUE=$(egrep ${VARNAME} ${PROJECT_ROOT}/.env | sed s/${VARNAME}=//) - eval "${VARNAME}=${ENV_VALUE:-$DEFAULT_VALUE}" -} - - -PROJECT_ROOT="./" - -# Default flags values. -get_default_value DEFAULT_DRUSH_ALIAS sitename.test -get_default_value DOCKER_PROJECT_ROOT /var/www/html -get_default_value NO_ACTION false -get_default_value DATABASE_ONLY false -get_default_value NO_DATABASE false -get_default_value REFRESH_LOCAL_DUMP false -get_default_value NPM_RUN_COMMAND dev -get_default_value UNCOMPRESSED false - -# Set the target remote Environment to download database. -DEFAULT_SITE=$(echo $DEFAULT_DRUSH_ALIAS | cut -d . -f 1) -REMOTE_ENVIRONMENT=$(echo $DEFAULT_DRUSH_ALIAS | cut -d . -f 2) - -SITE=$DEFAULT_SITE - -# Scripts/executables.compilation -DOCKER_EXEC_PHP="ddev exec" -DOCKER_EXEC_TTY_PHP="ddev exec" -DOCKER_EXEC_NPM="ddev exec" -COMPOSER_EXEC="ddev exec composer" -RM_EXEC="rm" -MAKE_EXEC="ddev" - -# Having a month based db backup filename ensures the database is refreshed at least every month. -BACKUP_FILE_NAME_TEMPLATE=db-$(date +%Y-%m) - -function show_help() { -cat << EOF - -Reloads a site's local environment with data from a given remote environment. - -Reloads includes: - - Execute composer install. - - Download database from remote environment and site (if needed). - - Imports downloaded database in local environment and site. - - Drupal update (updb and cim). - - Frontend assets generation (./scripts/frontend-build.sh) - -Usage: ${0##*/} [-d|--database-only] [-e|--env=(ENVIRONMENT_NAME)] [-s|--site=(SITE_NAME)] - -h|--help Show this help and exit. - - -d - --database-only Perfom only database sync operation (composer install and site update is not done). - - --no-database Do not perform database sync operation. - - -e=ENV - --env=ENV The environment from which to syncronize the database, as expresed in your drush aliases. - - -s=SITE - --site=SITE The site to reload, as expresed in your drush aliases. - - -r - --refresh Refresh the local dump file before importing it. This is always done if no local file is found. - - -n - --no-action Show actions that would be done but do not execute any command. Useful for debugging purposes. - - -u - --uncompressed Saves and tries to load the uncompressed backup - -You can add default values to most of the parameters by editing the .env file. -Here is a relation of the supported variables and their default values - -NO_ACTION=false -DATABASE_ONLY=false -NO_DATABASE=false -REFRESH_LOCAL_DUMP=false -DEFAULT_DRUSH_ALIAS=sitename.test -DOCKER_PROJECT_ROOT=/var/www/html -NPM_RUN_COMMAND=dev -UNCOMPRESSED=false - -If FRONTEND_THEME is not defined in the .env file frontend build step will be skipped. - -EOF -} - -####################################### -# Create database if not exist and grant privileges. -# Globals: -# PROJECT_ROOT -# SITE -# Arguments: -# None -# Returns: -# None -####################################### -function create_database() { - # Get connection credentials from .env file. - MYSQL_ROOT_PASS=$(egrep DB_ROOT_PASSWORD[^a-zA-Z_] ${PROJECT_ROOT}/.env | sed s/DB_ROOT_PASSWORD=//) - MYSQL_HOST=$(egrep DB_HOST[^a-zA-Z_] ${PROJECT_ROOT}/.env | sed s/DB_HOST=//) - MYSQL_DB_NAME=$(egrep DB_NAME[^a-zA-Z_] ${PROJECT_ROOT}/.env | sed s/DB_NAME=//) - MYSQL_DB_USER=$(egrep DB_USER[^a-zA-Z_] ${PROJECT_ROOT}/.env | sed s/DB_USER=//) - - if [ $SITE != $DEFAULT_SITE ] - then - MYSQL_DB_NAME="${MYSQL_DB_NAME}_${SITE}" - fi - - # Create DB. - $DOCKER_EXEC_PHP mysql -u root -p${MYSQL_ROOT_PASS} -h ${MYSQL_HOST} -e "CREATE DATABASE IF NOT EXISTS ${MYSQL_DB_NAME};GRANT ALL PRIVILEGES ON ${MYSQL_DB_NAME}.* TO '${MYSQL_DB_USER}'@'%';" -} - -# Process script options. -######################### -for i in "$@" -do - case ${i} in - -h|--help) - show_help # Display a usage synopsis. - exit - ;; - -d|--database-only) - DATABASE_ONLY=true - ;; - --no-database) - NO_DATABASE=true - ;; - -e=*|--env=*) # Takes an option argument; ensure it has been specified. - REMOTE_ENVIRONMENT="${i#*=}" - ;; - -s=*|--site=*) # Takes an option argument; ensure it has been specified. - SITE="${i#*=}" - ;; - -r|--refresh) - REFRESH_LOCAL_DUMP=true - ;; - -u|--uncompressed) - UNCOMPRESSED=true - ;; - -n|--no-action) - NO_ACTION=true - ;; - --) # End of all options. - shift - break - ;; - -?*|*) - printf 'ERROR: Unknown option: %s\n' "$1" >&2 - show_help - exit 1 - ;; - esac -shift -done - - -# Perform some sanity checks. -############################# - -if [ ${NO_DATABASE} = true ] && [ ${DATABASE_ONLY} = true ] -then - printf 'ERROR: The options --database-only and --no-database are not compatible.\n' "$1" >&2 - exit 1 -fi - -# Validate the site and environment. -VALID_ALIASES=$($DOCKER_EXEC_PHP drush sa --format list) -VALID_SITES=$(echo "${VALID_ALIASES[@]//@}" | cut -d . -f 1 | uniq) -VALID_ENVIRONMENTS=$(echo "${VALID_ALIASES[@]//@}" | cut -d . -f 2 | uniq) - -IS_VALID_ENVIRONMENT=$(echo "${VALID_ENVIRONMENTS[@]}" | grep -o "${REMOTE_ENVIRONMENT}" | wc -w) -if [ "${IS_VALID_ENVIRONMENT}" = 0 ] -then - echo "ERROR: Wrong environment: ${REMOTE_ENVIRONMENT}. Valid environments are:" - echo "${VALID_ENVIRONMENTS[@]}" - exit 1 -fi - -IS_VALID_SITE=$(echo "${VALID_SITES[@]}" | grep -o "${SITE}" | wc -w) -if [ "${IS_VALID_SITE}" = 0 ] -then - echo "ERROR: Wrong site: ${SITE}. Valid sites are:" - echo "${VALID_SITES[@]}" - exit 1 -fi - -echo "Reloading local environment ${SITE} site with database from $REMOTE_ENVIRONMENT remote environment." - -# Setup some calculated constants. -if [ $NO_ACTION = true ] -then - DOCKER_EXEC_PHP="echo $DOCKER_EXEC_PHP" - DOCKER_EXEC_TTY_PHP="echo $DOCKER_EXEC_TTY_PHP" - COMPOSER_EXEC="echo $COMPOSER_EXEC" - RM_EXEC="echo $RM_EXEC" - DOCKER_EXEC_NPM="echo $DOCKER_EXEC_NPM" - MAKE_EXEC="echo $MAKE_EXEC" -fi - -if [ $SITE = $DEFAULT_SITE ] -then - LOCAL_ALIAS="self" -else - LOCAL_ALIAS="$SITE.local" -fi - -REMOTE_ALIAS="$SITE.$REMOTE_ENVIRONMENT" - -if [[ ${UNCOMPRESSED} = false ]] -then - BACKUP_FILE_NAME=${BACKUP_FILE_NAME_TEMPLATE}.${SITE}.${REMOTE_ENVIRONMENT}.sql.gz -else - BACKUP_FILE_NAME=${BACKUP_FILE_NAME_TEMPLATE}.${SITE}.${REMOTE_ENVIRONMENT}.sql -fi - -LOCAL_FILE=${PROJECT_ROOT}tmp/${BACKUP_FILE_NAME} - -cd ${PROJECT_ROOT} - -if [[ ${DATABASE_ONLY} = false ]] -then - # Install dependencies. - $COMPOSER_EXEC install - -fi - -if [[ ${NO_DATABASE} = false ]] -then - - create_database - - # Drop current database. - $DOCKER_EXEC_PHP drush @${LOCAL_ALIAS} sql-drop -y - - # Do we need to download a remote dump? - # Best if we do this before droping current database. - if [ ${REFRESH_LOCAL_DUMP} = true ] || [ ! -f $LOCAL_FILE ] - then - echo "Either no local dump was found or refresh local dump options was passed." - # Ensure using your personal ssh-key before trying to connect to remote alias for downloading the database. - ssh-add -k - $DOCKER_EXEC_PHP drush sql:sync @${REMOTE_ALIAS} @${LOCAL_ALIAS} -y - $DOCKER_EXEC_PHP drush @${LOCAL_ALIAS} sql:sanitize -y - - if [ $? -ne 0 ] - then - echo "There was an error downloading the remote DB dump." - exit 1 - fi - else - echo "Loading database from local file: ${LOCAL_FILE}" - if [[ ${UNCOMPRESSED} = false ]] - then - zcat ${LOCAL_FILE} | $DOCKER_EXEC_TTY_PHP drush @${LOCAL_ALIAS} sql-cli - else - cat ${LOCAL_FILE} | $DOCKER_EXEC_TTY_PHP drush @${LOCAL_ALIAS} sql-cli - fi - fi - -fi - -if [[ ${DATABASE_ONLY} = false ]] -then - - # Execute updates and import configuration. - $DOCKER_EXEC_PHP drush @${LOCAL_ALIAS} updb -y - - $DOCKER_EXEC_PHP drush @${LOCAL_ALIAS} cim -y - - $DOCKER_EXEC_PHP drush @${LOCAL_ALIAS} deploy:hook -y - - $MAKE_EXEC frontend ${NPM_RUN_COMMAND} - - $DOCKER_EXEC_PHP drush @${LOCAL_ALIAS} cr - - # Show one-time login link. - $DOCKER_EXEC_PHP drush @${LOCAL_ALIAS} uli - -fi - -if [ ${REFRESH_LOCAL_DUMP} = true ] || [ ! -f $LOCAL_FILE ] - then - echo "Updating local dump." - if [[ ${UNCOMPRESSED} = false ]] - then - $DOCKER_EXEC_PHP drush @${LOCAL_ALIAS} sql:dump --gzip --result-file=../${LOCAL_FILE%.*} - else - $DOCKER_EXEC_PHP drush @${LOCAL_ALIAS} sql:dump --result-file=../${LOCAL_FILE} - fi -fi - -cat << EOF -////////////////////////////// -// RELOAD LOCAL COMPLETED // -////////////////////////////// -EOF \ No newline at end of file diff --git a/scripts/fix-adminer-norouter.sh b/scripts/fix-adminer-norouter.sh deleted file mode 100755 index 982395b..0000000 --- a/scripts/fix-adminer-norouter.sh +++ /dev/null @@ -1,10 +0,0 @@ -#ddev-generated -## This script overwrite the no-router adminer configuration with the default one, so when you run autoupdate -## on no-router environment, the adminer configuration is not overwritten. -pwd -cat < docker-compose.adminer_norouter.yaml -#ddev-generated -# If omit_containers[ddev-router] then this file will be replaced -# with another with a \`ports\` statement to directly expose port 8080 to 9100 -services: {} -EOF \ No newline at end of file diff --git a/tests/test.bats b/tests/test.bats index a231687..70e3d52 100644 --- a/tests/test.bats +++ b/tests/test.bats @@ -1,3 +1,5 @@ +#!/bin/bash + setup() { set -eu -o pipefail export DIR="$( cd "$( dirname "$BATS_TEST_FILENAME" )" >/dev/null 2>&1 && pwd )/.." @@ -18,23 +20,68 @@ teardown() { [ "${TESTDIR}" != "" ] && rm -rf ${TESTDIR} } +check_services() { + echo -n "Checking services:" + INSTALLED_SERVICES=$(ddev get --installed) + echo -n "Checking if Aljibe is installed..." + echo "$INSTALLED_SERVICES" | grep -q "aljibe" | echo " Ok." + echo -n "Checking if Aljibe Assistant is installed..." + echo "$INSTALLED_SERVICES" | grep -q "aljibe-assistant" | echo " Ok." + echo -n "Checking if Adminer is installed..." + echo "$INSTALLED_SERVICES" | grep -q "adminer" | echo " Ok." + echo -n "Checking if BackstopJS is installed..." + echo "$INSTALLED_SERVICES" | grep -q "backstopjs" | echo " Ok." + echo -n "Checking if lighthouse is installed..." + echo "$INSTALLED_SERVICES" | grep -q "lighthouse" | echo " Ok." + echo -n "Checking if mkdocs is installed..." + echo "$INSTALLED_SERVICES" | grep -q "mkdocs" | echo " Ok." + echo -n "Checking if pa11y is installed..." + echo "$INSTALLED_SERVICES" | grep -q "pa11y" | echo " Ok." + echo -n "Checking if redis is installed..." + echo "$INSTALLED_SERVICES" | grep -q "redis" | echo " Ok." + echo -n "Checking if selenium is installed..." + echo "$INSTALLED_SERVICES" | grep -q "selenium" | echo " Ok." + echo -n "Checking if unlighthouse is installed..." + echo "$INSTALLED_SERVICES" | grep -q "unlighthouse" | echo " Ok." + +} + +# Check if the project is browsable with wget +check_project_browse() { + echo -n "Checking if the project is browsable..." + curl -s https://${PROJNAME}.ddev.site | grep -q "Welcome" + echo " Ok." +} + +check_drupal_admin_access() { + echo -n "Checking if the Drupal admin is accessible..." + ddev drush uli + curl -sLb cookies $(ddev drush uli) | grep -q "The email address is not made public" + echo " Ok." +} + @test "install from directory" { set -eu -o pipefail cd ${TESTDIR} echo "# ddev get ${DIR} with project ${PROJNAME} in ${TESTDIR} ($(pwd))" >&3 ddev get ${DIR} - ddev restart - + ddev restart >/dev/null ddev aljibe-assistant --auto + check_services >&3 + check_project_browse >&3 + check_drupal_admin_access >&3 } -@test "install from release" { - set -eu -o pipefail - cd ${TESTDIR} || ( printf "unable to cd to ${TESTDIR}\n" && exit 1 ) - echo "# ddev get drud/ddev-addon-template with project ${PROJNAME} in ${TESTDIR} ($(pwd))" >&3 - ddev get drud/ddev-addon-template - ddev restart >/dev/null - # Do something useful here that verifies the add-on - # ddev exec "curl -s elasticsearch:9200" | grep "${PROJNAME}-elasticsearch" -} +#@test "install from release" { +# set -eu -o pipefail +# cd ${TESTDIR} || ( printf "unable to cd to ${TESTDIR}\n" && exit 1 ) +# echo "# ddev get metadrop/ddev-aljibe with project ${PROJNAME} in ${TESTDIR} ($(pwd))" >&3 +# ddev get metadrop/ddev-aljibe +# ddev restart >/dev/null +# ddev aljibe-assistant --auto +# +# check_services >&3 +# check_project_browse >&3 +# check_drupal_admin_access >&3 +#}