#!/bin/bash PROGRAM_NAME=$(basename "$0") ###### Utility functions yml_pattern_exists() { PATTERN=$1 if ! type yq > /dev/null 2>&1; then error "yq program is not installed" fi if [ -f "$ABRA_CONFIG" ]; then RESULT=$(yq read "$ABRA_CONFIG" "$PATTERN") if [ "$RESULT" != 0 ]; then return 0 fi fi return 1 } parse_subcommand() { SUBCOMMAND="$1" PREFIX=$2 if [ -n "$PREFIX" ]; then PPREFIX="_$2" SPREFIX="$2 " SSPREFIX=" $2" fi case $SUBCOMMAND in "" | "-h" | "--help") "sub${PPREFIX}_help" ;; *) shift 2 "sub${PPREFIX}_${SUBCOMMAND}" "$@" if [ $? = 127 ]; then echo "Error: '$SPREFIX$SUBCOMMAND' is not a known subcommand." >&2 echo " Run '$PROGRAM_NAME$SSPREFIX --help' for a list of known subcommands." >&2 exit 1 fi ;; esac } error() { echo "$(tput setaf 1)ERROR: $*$(tput sgr0)" exit 1 } warning() { echo "$(tput setaf 3)WARNING: $*$(tput sgr0)" } ###### Top-level arguments ABRA_CONFIG=abra.yml if [ "$1" == "-c" ]; then ABRA_CONFIG=$2 shift 2 fi if [ "$1" == "-e" ]; then ABRA_ENV=$2 shift 2 fi if [ "$1" == "-a" ]; then STACK_NAME=$2 shift 2 fi ###### Load config if [ -f "$ABRA_CONFIG" ]; then if yml_pattern_exists stack_name; then STACK_NAME=$(yq read "$ABRA_CONFIG" stack_name) fi # FIXME load other variables somehow fi if [ -n "$ABRA_ENV" ]; then # shellcheck disable=SC1090 source "$ABRA_ENV" fi ###### Default settings if [ -z "$COMPOSE_FILE" ]; then COMPOSE_FILE="compose.yml" fi if [ -z "$ABRA_STACK_DIR" ]; then ABRA_STACK_DIR="stacks/$SERVICE" fi load_context() { if [ -z "$DOCKER_CONTEXT" ]; then warning "\$DOCKER_CONTEXT not set, (slowly) looking it up" DOCKER_CONTEXT=$(docker context ls | grep '*' | cut -d' ' -f1) fi } ###### Safety checks require_stack() { if [ -z "$STACK_NAME" ]; then error "no stack_name, export \$STACK_NAME=my_cool_app or add it to abra.yml" fi } require_stack_dir() { if [ -z "$ABRA_STACK_DIR" ] || [ ! -d "$ABRA_STACK_DIR" ]; then error "can't find \$ABRA_STACK_DIR '$ABRA_STACK_DIR'" fi } if [ -z "$ABRA_ENV" ] && [ -f .envrc ] && type direnv > /dev/null 2>&1 && ! direnv status | grep -q 'Found RC allowed true'; then error "direnv is blocked, run direnv allow" fi ###### Custom commands if [ -f abra-commands.sh ]; then # shellcheck disable=SC1091 source abra-commands.sh fi ###### Global help sub_help() { echo "Usage: $PROGRAM_NAME [-a STACK_NAME] [-c CONFIG] [-e ENV_FILE] [options]" echo "" echo "Subcommands:" echo " context [--help] [SUBCOMMAND] manage remote swarm contexts" echo " cp SRC_PATH SERVICE:DEST_PATH copy files to a container" echo " deploy let 'em rip" echo " logs SERVICE [ARGS] tail logs from a deployed service" echo " run SERVICE CMD run a command in the specified service's container" echo " run_args SERVICE ARGS CMD run, passing extra args to docker exec" echo " secret [--help] [SUBCOMMAND] manage secrets" echo " upgrade upgrade to the latest version" echo " ... (custom commands)" echo "" echo "Make sure \$STACK_NAME is set using direnv, -a, -e or -c" echo "" echo "Runs compose.yml by default, set e.g. COMPOSE_FILE=\"compose.yml:compose2.yml\" to override" } ###### Subcommand `secret` sub_secret_help() { echo "Usage: $PROGRAM_NAME [-a STACK_NAME] secret [options]" echo "" echo "Subcommands:" echo " generate SECRET VERSION [PW] generate & store secret" } sub_secret_generate(){ require_stack load_context SECRET=$1 VERSION=$2 PW=${3:-pwqgen} if [ -z "$SECRET" ] || [ -z "$VERSION" ]; then echo "Usage: $PROGRAM_NAME secret_generate SECRET VERSION" exit fi $PW | tee \ >(docker secret create "${STACK_NAME}_${SECRET}_${VERSION}" -) \ >(pass insert "hosts/$DOCKER_CONTEXT/${STACK_NAME}/${SECRET}" -m) } sub_secret() { SUBCOMMAND=$1 shift # shellcheck disable=SC2068 parse_subcommand "$SUBCOMMAND" "secret" $@ } ###### Subcommand `run` sub_run_args(){ require_stack SERVICE=$1 DOCKER_ARGS=$2 shift 2 if [ -z "$SERVICE" ]; then echo "Usage: $PROGRAM_NAME run SERVICE [CMD]" exit fi CONTAINER=$(docker container ls --format "table {{.ID}},{{.Names}}" \ | grep "${STACK_NAME}_${SERVICE}" | cut -d',' -f1) if [ -z "$CONTAINER" ]; then error "Can't find a container for ${STACK_NAME}_${SERVICE}" exit fi # shellcheck disable=SC2086 docker exec $DOCKER_ARGS -it "$CONTAINER" "$@" return } sub_run(){ SERVICE=$1 shift sub_run_args "$SERVICE" "" "$@" } ###### Subcommand `deploy` sub_deploy (){ require_stack require_stack_dir load_context echo "About to deploy:" echo " Context: $(tput setaf 4)${DOCKER_CONTEXT}$(tput sgr0)" echo " Compose: $(tput setaf 3)${ABRA_STACK_DIR}/${COMPOSE_FILE}$(tput sgr0)" if [ -n "$DOMAIN" ]; then echo " Domain: $(tput setaf 2)${DOMAIN}$(tput sgr0)" fi echo " Stack: $(tput setaf 1)${STACK_NAME}$(tput sgr0)" read -rp "Continue? (y/[n])? " choice case "$choice" in y|Y ) ;; n|N ) return;; * ) return;; esac ( cd "$ABRA_STACK_DIR" || error "\$ABRA_STACK_DIR '$ABRA_STACK_DIR' not found" # shellcheck disable=SC2086 if docker stack deploy -c ${COMPOSE_FILE/:/ -c } "$STACK_NAME"; then if [ -n "$DOMAIN" ]; then echo "$(tput setaf 2)Yay! App should be available at https://${DOMAIN}$(tput sgr0)" else echo "$(tput setaf 2)Yay! That worked. No \$DOMAIN defined, check logs.(tput sgr0)" fi else error "Oh no! Something went wrong 😕 Check errors above" fi ) } ###### Subcommand `logs` sub_logs (){ require_stack SERVICE=$1 shift if [ $# -eq 0 ]; then LOGS_ARGS="\ --follow \ --no-trunc \ --details \ --timestamps" else # shellcheck disable=SC2124 LOGS_ARGS=$@ fi # shellcheck disable=SC2086 docker service logs "${STACK_NAME}_${SERVICE}" $LOGS_ARGS } ###### Subcommand `cp` sub_cp() { require_stack SOURCE=$1 DEST=$2 SERVICE=$(echo "$SOURCE" | grep -o '^[^:]\+:' || echo "$DEST" | grep -o '^[^:]\+:') SERVICE=$(echo "$SERVICE" | tr -d ':') if [ -z "$SERVICE" ]; then echo "Usage: $PROGRAM_NAME cp SERVICE:SRC_PATH DEST_PATH" echo " $PROGRAM_NAME cp SRC_PATH SERVICE:DEST_PATH" echo "" error "Can't find SERVICE in either SRC or DEST" fi CONTAINER=$(docker container ls --format "table {{.ID}},{{.Names}}" \ | grep "${STACK_NAME}_${SERVICE}" | cut -d',' -f1) if [ -z "$CONTAINER" ]; then error "Can't find a container for ${STACK_NAME}_${SERVICE}" exit fi CP_ARGS=$(echo "$SOURCE $DEST" | sed "s/$SERVICE/$CONTAINER/") # shellcheck disable=SC2086 docker cp $CP_ARGS } ###### Subcommand `context` sub_context_help() { echo "Usage: $PROGRAM_NAME [-a STACK_NAME] context [options]" echo "" echo "Subcommands:" echo " init HOST [USER] [PORT] set up remote Docker context" echo " use activate remote Docker context" } sub_context_init() { HOST="$1" USERNAME="$2" PORT="$3" if [ -n "$PORT" ]; then PORT=":$PORT" fi if [ -n "$USERNAME" ]; then USERNAME="$USERNAME@" fi docker context create "$HOST" \ --docker "host=ssh://$USERNAME$HOST$PORT" } sub_context_use() { docker context use "$1" } sub_context() { SUBCOMMAND2=$1 shift # shellcheck disable=SC2068 parse_subcommand "$SUBCOMMAND2" "context" $@ } ###### Subcommand `upgrade` sub_upgrade() { curl -fsSL https://install.abra.autonomic.zone | bash } ###### Main SUBCOMMAND=$1 shift # shellcheck disable=SC2086,SC2068 parse_subcommand $SUBCOMMAND "" $@