#!/usr/bin/env bash set -eo pipefail; [[ $DOKKU_TRACE ]] && set -x source "$(dirname "$0")/config" get_random_port() { local port=$RANDOM local quit=0 while [ "$quit" -ne 1 ]; do netstat -a | grep $port >> /dev/null if [ $? -gt 0 ]; then quit=1 else port=$((port + 1)) fi done echo $port } get_container_ip() { docker inspect --format '{{ .NetworkSettings.IPAddress }}' "$1" } verify_service_name() { local SERVICE="$1" [[ ! -n "$SERVICE" ]] && dokku_log_fail "(verify_service_name) SERVICE must not be null" [[ ! -d "$PLUGIN_DATA_ROOT/$SERVICE" ]] && dokku_log_fail "$PLUGIN_SERVICE service $SERVICE does not exist" return 0 } service_alias() { local SERVICE="$1" local ALIAS_FILE="$PLUGIN_DATA_ROOT/$SERVICE/ALIAS" verify_service_name "$1" if [[ -f "$ALIAS_FILE" ]]; then cat "$ALIAS_FILE" else echo "$PLUGIN_DEFAULT_ALIAS" fi } service_info() { local SERVICE="$1" local SERVICE_URL=$(service_url "$SERVICE") echo " DSN: $SERVICE_URL" } service_list() { local SERVICES=$(ls $PLUGIN_DATA_ROOT 2> /dev/null) if [[ -z $SERVICES ]]; then dokku_log_warn "There are no $PLUGIN_SERVICE services" else dokku_log_info1_quiet "$PLUGIN_SERVICE services:" for SERVICE in $SERVICES; do dokku_log_verbose "$SERVICE $(service_status $SERVICE)" done fi } service_link() { local APP="$2" local SERVICE="$1" local SERVICE_ROOT="$PLUGIN_DATA_ROOT/$SERVICE" local LINKS_FILE="$SERVICE_ROOT/LINKS" mkdir -p "$SERVICE_ROOT" || dokku_log_fail "Unable to create service directory" touch "$LINKS_FILE" echo "$APP" >> "$LINKS_FILE" sort "$LINKS_FILE" -u -o "$LINKS_FILE" dokku_log_info1 "Restarting app $APP" dokku ps:restart "$APP" } service_logs() { local SERVICE="$1" local SERVICE_ROOT="$PLUGIN_DATA_ROOT/$SERVICE" local ID=$(cat "$SERVICE_ROOT/ID") if [[ $2 == "-t" ]]; then DOKKU_LOGS_ARGS="--follow" else DOKKU_LOGS_ARGS="--tail 100" fi docker logs $DOKKU_LOGS_ARGS "$ID" } service_set_alias() { local SERVICE="$1" local SERVICE_ROOT="$PLUGIN_DATA_ROOT/$SERVICE" local ALIAS_FILE="$SERVICE_ROOT/ALIAS" mkdir -p "$SERVICE_ROOT" || dokku_log_fail "Unable to create service directory" touch "$ALIAS_FILE" echo "$2" > "$ALIAS_FILE" } service_status() { local SERVICE="$1" local SERVICE_ROOT="$PLUGIN_DATA_ROOT/$SERVICE" local ID="$(cat "$SERVICE_ROOT/ID")" is_container_status "$ID" "Dead" && echo "(dead)" && return 0 is_container_status "$ID" "OOMKilled" && echo "(oomkilled)" && return 0 is_container_status "$ID" "Paused" && echo "(paused)" && return 0 is_container_status "$ID" "Restarting" && echo "(restarting)" && return 0 is_container_status "$ID" "Running" && echo "(running)" && return 0 echo "(stopped)" && return 0 } service_port_expose() { service_port_unpause "$1" "true" } service_port_pause() { local SERVICE="$1" local SERVICE_ROOT="$PLUGIN_DATA_ROOT/$SERVICE" local PORT_FILE="$SERVICE_ROOT/PORT" local DESTINATION_FILE="$SERVICE_ROOT/IPTABLES_DESTINATION" local LOG_FAIL="$2" if [[ "$LOG_FAIL" == "true" ]]; then [[ ! -f "$PORT_FILE" ]] && dokku_log_fail "Service not exposed" else [[ ! -f "$PORT_FILE" ]] && return 0 fi local ID=$(cat "$SERVICE_ROOT/ID") local IP=$(get_container_ip "$ID") local PORT=$(cat "$PORT_FILE") local DESTINATION=$(cat "$DESTINATION_FILE") sudo /sbin/iptables -t nat -D DOCKER -p tcp --dport "$PORT" -j DNAT --to-destination "$DESTINATION" } service_port_unexpose() { service_port_pause "$1" "true" rm -rf "$PORT_FILE" } service_port_unpause() { local SERVICE="$1" local SERVICE_ROOT="$PLUGIN_DATA_ROOT/$SERVICE" local PORT_FILE="$SERVICE_ROOT/PORT" local DESTINATION_FILE="$SERVICE_ROOT/IPTABLES_DESTINATION" local LOG_FAIL="$2" local PORT=$(get_random_port) local ID=$(cat "$SERVICE_ROOT/ID") local IP=$(get_container_ip "$ID") local DESTINATION="$IP:$PLUGIN_DATASTORE_PORT" if [[ "$LOG_FAIL" == "true" ]]; then [[ -f "$PORT_FILE" ]] && PORT=$(cat "$PORT_FILE") && dokku_log_fail "Service $SERVICE already exposed on port $PORT" else [[ ! -f "$PORT_FILE" ]] && return 0 PORT=$(cat "$PORT_FILE") && sudo /sbin/iptables -t nat -D DOCKER -p tcp --dport "$PORT" -j DNAT --to-destination "$DESTINATION" fi echo "$PORT" > "$PORT_FILE" echo "$DESTINATION" > "$DESTINATION_FILE" echo "$DESTINATION" sudo /sbin/iptables -t nat -A DOCKER -p tcp --dport "$PORT" -j DNAT --to-destination "$DESTINATION" if [[ "$LOG_FAIL" == "true" ]]; then dokku_log_info1 "Service $SERVICE exposed on port $PORT" fi } service_start() { local SERVICE="$1" local SERVICE_ROOT="$PLUGIN_DATA_ROOT/$SERVICE" dokku_log_verbose_quiet "Starting container" if [[ -f "$SERVICE_ROOT/ID" ]] && docker ps -aq --no-trunc | grep -q $(cat "$SERVICE_ROOT/ID"); then ID=$(cat "$SERVICE_ROOT/ID") docker start "$ID" > /dev/null service_port_unpause "$SERVICE" dokku_log_info2 "Container started" else dokku_log_verbose_quiet "No container exists for $SERVICE" fi } service_stop() { local SERVICE="$1" local SERVICE_ROOT="$PLUGIN_DATA_ROOT/$SERVICE"; if [[ -f "$SERVICE_ROOT/ID" ]] && docker ps -aq --no-trunc | grep -q $(cat "$SERVICE_ROOT/ID"); then dokku_log_verbose_quiet "Stopping container" ID=$(cat "$SERVICE_ROOT/ID") docker stop "$ID" > /dev/null service_port_pause "$SERVICE" dokku_log_info2 "Container stopped" else dokku_log_verbose_quiet "No container exists for $SERVICE" fi } service_unlink() { local APP="$2" local SERVICE="$1" local SERVICE_ROOT="$PLUGIN_DATA_ROOT/$SERVICE" local LINKS_FILE="$SERVICE_ROOT/LINKS" mkdir -p "$SERVICE_ROOT" || dokku_log_fail "Unable to create service directory" touch "$LINKS_FILE" sed -i "/^$APP\$/d" "$LINKS_FILE" sort "$LINKS_FILE" -u -o "$LINKS_FILE" dokku_log_info1 "Restarting app $APP" dokku ps:restart "$APP" } service_url() { local SERVICE="$1" local SERVICE_ROOT="$PLUGIN_DATA_ROOT/$SERVICE" local ID="$(cat "$SERVICE_ROOT/ID")" local IP="$(get_container_ip "$ID")" echo "$PLUGIN_SCHEME://$IP:$PLUGIN_DATASTORE_PORT/0" } is_container_status () { local CID=$1 local TEMPLATE="{{.State.$2}}" local CONTAINER_STATUS=$(docker inspect -f "$TEMPLATE" "$CID" || true) if [[ "$CONTAINER_STATUS" == "true" ]]; then return 0 else return 1 fi }