Files
bootstrap/tools/portainer/stack-redeploy.sh
2026-02-22 17:52:23 +00:00

184 lines
5.3 KiB
Bash
Executable File

#!/usr/bin/env bash
#
# stack-redeploy.sh - Redeploy a Portainer stack
#
# For git-based stacks, this pulls the latest from the repository and redeploys.
# For file-based stacks, this redeploys with the current stack file.
#
# Usage: stack-redeploy.sh -n <stack-name> [-p] [-e endpoint_id]
#
# Environment variables:
# PORTAINER_URL - Portainer instance URL (e.g., https://portainer.example.com:9443)
# PORTAINER_API_KEY - API access token
#
# Options:
# -n name Stack name (required)
# -i id Stack ID (alternative to -n)
# -p Pull latest images before redeploying
# -e endpoint_id Endpoint/environment ID (auto-detected from stack if not provided)
# -h Show this help
set -euo pipefail
# Default values
STACK_NAME=""
STACK_ID=""
PULL_IMAGE=false
ENDPOINT_ID=""
# Parse arguments
while getopts "n:i:pe:h" opt; do
case $opt in
n) STACK_NAME="$OPTARG" ;;
i) STACK_ID="$OPTARG" ;;
p) PULL_IMAGE=true ;;
e) ENDPOINT_ID="$OPTARG" ;;
h)
head -22 "$0" | grep "^#" | sed 's/^# \?//'
exit 0
;;
*)
echo "Usage: $0 -n <stack-name> [-p] [-e endpoint_id]" >&2
exit 1
;;
esac
done
# Validate environment
if [[ -z "${PORTAINER_URL:-}" ]]; then
echo "Error: PORTAINER_URL environment variable not set" >&2
exit 1
fi
if [[ -z "${PORTAINER_API_KEY:-}" ]]; then
echo "Error: PORTAINER_API_KEY environment variable not set" >&2
exit 1
fi
if [[ -z "$STACK_NAME" && -z "$STACK_ID" ]]; then
echo "Error: Either -n <stack-name> or -i <stack-id> is required" >&2
exit 1
fi
# Remove trailing slash from URL
PORTAINER_URL="${PORTAINER_URL%/}"
# Function to make API requests
api_request() {
local method="$1"
local endpoint="$2"
local data="${3:-}"
local args=(-s -w "\n%{http_code}" -X "$method" -H "X-API-Key: ${PORTAINER_API_KEY}")
if [[ -n "$data" ]]; then
args+=(-H "Content-Type: application/json" -d "$data")
fi
curl "${args[@]}" "${PORTAINER_URL}${endpoint}"
}
# Get stack info by name or ID
if [[ -n "$STACK_NAME" ]]; then
echo "Looking up stack '$STACK_NAME'..."
response=$(api_request GET "/api/stacks")
http_code=$(echo "$response" | tail -n1)
body=$(echo "$response" | sed '$d')
if [[ "$http_code" != "200" ]]; then
echo "Error: Failed to list stacks (HTTP $http_code)" >&2
exit 1
fi
stack_info=$(echo "$body" | jq --arg name "$STACK_NAME" '.[] | select(.Name == $name)')
if [[ -z "$stack_info" || "$stack_info" == "null" ]]; then
echo "Error: Stack '$STACK_NAME' not found" >&2
exit 1
fi
STACK_ID=$(echo "$stack_info" | jq -r '.Id')
ENDPOINT_ID_FROM_STACK=$(echo "$stack_info" | jq -r '.EndpointId')
else
# Get stack info by ID
response=$(api_request GET "/api/stacks/${STACK_ID}")
http_code=$(echo "$response" | tail -n1)
body=$(echo "$response" | sed '$d')
if [[ "$http_code" != "200" ]]; then
echo "Error: Failed to get stack (HTTP $http_code)" >&2
exit 1
fi
stack_info="$body"
STACK_NAME=$(echo "$stack_info" | jq -r '.Name')
ENDPOINT_ID_FROM_STACK=$(echo "$stack_info" | jq -r '.EndpointId')
fi
# Use endpoint ID from stack if not provided
if [[ -z "$ENDPOINT_ID" ]]; then
ENDPOINT_ID="$ENDPOINT_ID_FROM_STACK"
fi
# Check if this is a git-based stack
git_config=$(echo "$stack_info" | jq -r '.GitConfig // empty')
if [[ -n "$git_config" && "$git_config" != "null" ]]; then
echo "Stack '$STACK_NAME' (ID: $STACK_ID) is git-based"
echo "Triggering git pull and redeploy..."
# Git-based stack redeploy
# The git redeploy endpoint pulls from the repository and redeploys
request_body=$(jq -n \
--argjson pullImage "$PULL_IMAGE" \
'{
"pullImage": $pullImage,
"prune": false,
"repositoryReferenceName": "",
"repositoryAuthentication": false
}')
response=$(api_request PUT "/api/stacks/${STACK_ID}/git/redeploy?endpointId=${ENDPOINT_ID}" "$request_body")
else
echo "Stack '$STACK_NAME' (ID: $STACK_ID) is file-based"
# Get the current stack file content
echo "Fetching current stack file..."
response=$(api_request GET "/api/stacks/${STACK_ID}/file")
http_code=$(echo "$response" | tail -n1)
body=$(echo "$response" | sed '$d')
if [[ "$http_code" != "200" ]]; then
echo "Error: Failed to get stack file (HTTP $http_code)" >&2
exit 1
fi
stack_file_content=$(echo "$body" | jq -r '.StackFileContent')
echo "Redeploying..."
request_body=$(jq -n \
--argjson pullImage "$PULL_IMAGE" \
--arg stackFile "$stack_file_content" \
'{
"pullImage": $pullImage,
"prune": false,
"stackFileContent": $stackFile
}')
response=$(api_request PUT "/api/stacks/${STACK_ID}?endpointId=${ENDPOINT_ID}" "$request_body")
fi
http_code=$(echo "$response" | tail -n1)
body=$(echo "$response" | sed '$d')
if [[ "$http_code" == "200" ]]; then
echo "Successfully redeployed stack '$STACK_NAME'"
if [[ "$PULL_IMAGE" == "true" ]]; then
echo " - Pulled latest images"
fi
else
echo "Error: Redeploy failed (HTTP $http_code)" >&2
echo "$body" | jq '.' 2>/dev/null || echo "$body" >&2
exit 1
fi