425 lines
8.8 KiB
Bash
Executable File
425 lines
8.8 KiB
Bash
Executable File
#!/bin/bash
|
|
|
|
PROGRAM_NAME=$(basename "$0")
|
|
|
|
###### Utility functions
|
|
|
|
yml_pattern_exists() {
|
|
PATTERN=$1
|
|
|
|
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)"
|
|
}
|
|
|
|
success() {
|
|
echo "$(tput setaf 2)$*$(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
|
|
require_yq
|
|
|
|
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" || error "Unable to load env from '$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"
|
|
# shellcheck disable=SC2063
|
|
DOCKER_CONTEXT=$(docker context ls | grep '*' | cut -d' ' -f1)
|
|
# FIXME 3wc: make sure grep doesn't parse this, we're want a literal '*'
|
|
fi
|
|
}
|
|
|
|
###### Safety checks
|
|
|
|
require_yq() {
|
|
if ! type yq > /dev/null 2>&1; then
|
|
error "yq program is not installed"
|
|
fi
|
|
}
|
|
|
|
require_multitail() {
|
|
if ! type multitail > /dev/null 2>&1; then
|
|
error "multitail program is not installed"
|
|
fi
|
|
}
|
|
|
|
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] <subcommand> [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 " multilogs tail logs from a whole stackk"
|
|
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 [global opts] secret <subcommand> [sub opts]"
|
|
echo ""
|
|
echo "Subcommands:"
|
|
echo " generate SECRET VERSION [PWGEN] generate & store secret"
|
|
echo " insert SECRET VERSION PW save PW in docker and pass"
|
|
}
|
|
|
|
sub_secret_insert() {
|
|
require_stack
|
|
load_context
|
|
|
|
SECRET=$1
|
|
VERSION=$2
|
|
PW=$3
|
|
|
|
if [ -z "$SECRET" ] || [ -z "$VERSION" ] || [ -z "$PW" ]; then
|
|
echo "Usage: $PROGRAM_NAME secret insert SECRET VERSION PW"
|
|
exit
|
|
fi
|
|
|
|
echo "$PW" | docker secret create "${STACK_NAME}_${SECRET}_${VERSION}" - > /dev/null
|
|
echo "$PW" | pass insert "hosts/$DOCKER_CONTEXT/${STACK_NAME}/${SECRET}" -m > /dev/null
|
|
}
|
|
|
|
sub_secret_generate(){
|
|
SECRET=$1
|
|
VERSION=$2
|
|
PWGEN=${3:-pwqgen}
|
|
|
|
PW=$($PWGEN)
|
|
|
|
success "Password: $PW"
|
|
|
|
echo "sub_secret_insert \"$SECRET\" \"$VERSION\" \"$PW\""
|
|
exit
|
|
sub_secret_insert "$SECRET" "$VERSION" "$PW"
|
|
}
|
|
|
|
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 DOCKER_ARGS [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
|
|
success "Yay! App should be available at https://${DOMAIN}"
|
|
else
|
|
success "Yay! That worked. No \$DOMAIN defined, check logs."
|
|
fi
|
|
else
|
|
error "Oh no! Something went wrong 😕 Check errors above"
|
|
fi
|
|
)
|
|
}
|
|
|
|
###### Subcommand `logs`
|
|
|
|
# Inspired by https://github.com/moby/moby/issues/31458#issuecomment-475411564
|
|
sub_multilogs() {
|
|
require_stack
|
|
require_multitail
|
|
|
|
# Get a list of the service names
|
|
SERVICES=$(docker stack services --format "{{.Name}}" "${STACK_NAME}")
|
|
# Sort the service names
|
|
SERVICES=$(echo "${SERVICES}" | sort)
|
|
# Create the command to run
|
|
COMMAND='multitail --mergeall'
|
|
for SERVICE in ${SERVICES}; do
|
|
COMMAND="${COMMAND} -L 'docker service logs --tail 20 -f ${SERVICE}'"
|
|
done
|
|
# Run the command
|
|
bash -c "${COMMAND}"
|
|
}
|
|
|
|
sub_logs (){
|
|
require_stack
|
|
|
|
SERVICE=$1
|
|
|
|
if [ -z "$SERVICE" ]; then
|
|
warning "No \$SERVICE provided, running multilogs"
|
|
sub_multilogs
|
|
fi
|
|
|
|
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 [global opts] context <subcommand> [sub opts]"
|
|
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 "" $@
|