chore: sync workflows from base-template
This commit is contained in:
51
.github/workflows/script/re-pull.sh
vendored
51
.github/workflows/script/re-pull.sh
vendored
@@ -5,8 +5,12 @@
|
|||||||
: "${PORTAINER_PASSWORD:?PORTAINER_PASSWORD tidak di-set}"
|
: "${PORTAINER_PASSWORD:?PORTAINER_PASSWORD tidak di-set}"
|
||||||
: "${STACK_NAME:?STACK_NAME tidak di-set}"
|
: "${STACK_NAME:?STACK_NAME tidak di-set}"
|
||||||
|
|
||||||
|
# Timeout total: MAX_RETRY * SLEEP_INTERVAL detik
|
||||||
|
MAX_RETRY=60 # 60 × 10s = 10 menit
|
||||||
|
SLEEP_INTERVAL=10
|
||||||
|
|
||||||
echo "🔐 Autentikasi ke Portainer..."
|
echo "🔐 Autentikasi ke Portainer..."
|
||||||
TOKEN=$(curl -s -X POST https://${PORTAINER_URL}/api/auth \
|
TOKEN=$(curl -s -X POST "https://${PORTAINER_URL}/api/auth" \
|
||||||
-H "Content-Type: application/json" \
|
-H "Content-Type: application/json" \
|
||||||
-d "{\"username\": \"${PORTAINER_USERNAME}\", \"password\": \"${PORTAINER_PASSWORD}\"}" \
|
-d "{\"username\": \"${PORTAINER_USERNAME}\", \"password\": \"${PORTAINER_PASSWORD}\"}" \
|
||||||
| jq -r .jwt)
|
| jq -r .jwt)
|
||||||
@@ -17,13 +21,12 @@ if [ -z "$TOKEN" ] || [ "$TOKEN" = "null" ]; then
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
echo "🔍 Mencari stack: $STACK_NAME..."
|
echo "🔍 Mencari stack: $STACK_NAME..."
|
||||||
STACK=$(curl -s -X GET https://${PORTAINER_URL}/api/stacks \
|
STACK=$(curl -s -X GET "https://${PORTAINER_URL}/api/stacks" \
|
||||||
-H "Authorization: Bearer ${TOKEN}" \
|
-H "Authorization: Bearer ${TOKEN}" \
|
||||||
| jq ".[] | select(.Name == \"$STACK_NAME\")")
|
| jq ".[] | select(.Name == \"$STACK_NAME\")")
|
||||||
|
|
||||||
if [ -z "$STACK" ]; then
|
if [ -z "$STACK" ]; then
|
||||||
echo "❌ Stack '$STACK_NAME' tidak ditemukan di Portainer!"
|
echo "❌ Stack '$STACK_NAME' tidak ditemukan di Portainer!"
|
||||||
echo " Pastikan nama stack sudah benar."
|
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -31,6 +34,16 @@ STACK_ID=$(echo "$STACK" | jq -r .Id)
|
|||||||
ENDPOINT_ID=$(echo "$STACK" | jq -r .EndpointId)
|
ENDPOINT_ID=$(echo "$STACK" | jq -r .EndpointId)
|
||||||
ENV=$(echo "$STACK" | jq '.Env // []')
|
ENV=$(echo "$STACK" | jq '.Env // []')
|
||||||
|
|
||||||
|
# ── Catat container ID lama sebelum redeploy ──────────────────────────────────
|
||||||
|
echo "📸 Mencatat container aktif sebelum redeploy..."
|
||||||
|
CONTAINERS_BEFORE=$(curl -s -X GET \
|
||||||
|
"https://${PORTAINER_URL}/api/endpoints/${ENDPOINT_ID}/docker/containers/json?all=true&filters=%7B%22label%22%3A%5B%22com.docker.compose.project%3D${STACK_NAME}%22%5D%7D" \
|
||||||
|
-H "Authorization: Bearer ${TOKEN}")
|
||||||
|
|
||||||
|
OLD_IDS=$(echo "$CONTAINERS_BEFORE" | jq -r '[.[] | .Id] | join(",")')
|
||||||
|
echo " Container lama: $(echo "$CONTAINERS_BEFORE" | jq -r '[.[] | .Names[0]] | join(", ")')"
|
||||||
|
|
||||||
|
# ── Ambil compose file lalu trigger redeploy ─────────────────────────────────
|
||||||
echo "📄 Mengambil compose file..."
|
echo "📄 Mengambil compose file..."
|
||||||
STACK_FILE=$(curl -s -X GET "https://${PORTAINER_URL}/api/stacks/${STACK_ID}/file" \
|
STACK_FILE=$(curl -s -X GET "https://${PORTAINER_URL}/api/stacks/${STACK_ID}/file" \
|
||||||
-H "Authorization: Bearer ${TOKEN}" \
|
-H "Authorization: Bearer ${TOKEN}" \
|
||||||
@@ -41,7 +54,7 @@ PAYLOAD=$(jq -n \
|
|||||||
--argjson env "$ENV" \
|
--argjson env "$ENV" \
|
||||||
'{stackFileContent: $content, env: $env, pullImage: true}')
|
'{stackFileContent: $content, env: $env, pullImage: true}')
|
||||||
|
|
||||||
echo "🚀 Redeploying $STACK_NAME (pull latest image)..."
|
echo "🚀 Triggering redeploy $STACK_NAME (pull latest image)..."
|
||||||
HTTP_STATUS=$(curl -s -o /tmp/portainer_response.json -w "%{http_code}" \
|
HTTP_STATUS=$(curl -s -o /tmp/portainer_response.json -w "%{http_code}" \
|
||||||
-X PUT "https://${PORTAINER_URL}/api/stacks/${STACK_ID}?endpointId=${ENDPOINT_ID}" \
|
-X PUT "https://${PORTAINER_URL}/api/stacks/${STACK_ID}?endpointId=${ENDPOINT_ID}" \
|
||||||
-H "Authorization: Bearer ${TOKEN}" \
|
-H "Authorization: Bearer ${TOKEN}" \
|
||||||
@@ -54,40 +67,44 @@ if [ "$HTTP_STATUS" != "200" ]; then
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
echo "⏳ Menunggu container running..."
|
echo "⏳ Menunggu image selesai di-pull dan container baru running..."
|
||||||
|
echo " (Timeout: $((MAX_RETRY * SLEEP_INTERVAL)) detik)"
|
||||||
|
|
||||||
MAX_RETRY=15
|
|
||||||
COUNT=0
|
COUNT=0
|
||||||
|
|
||||||
while [ $COUNT -lt $MAX_RETRY ]; do
|
while [ $COUNT -lt $MAX_RETRY ]; do
|
||||||
sleep 5
|
sleep $SLEEP_INTERVAL
|
||||||
COUNT=$((COUNT + 1))
|
COUNT=$((COUNT + 1))
|
||||||
|
|
||||||
CONTAINERS=$(curl -s -X GET \
|
CONTAINERS=$(curl -s -X GET \
|
||||||
"https://${PORTAINER_URL}/api/endpoints/${ENDPOINT_ID}/docker/containers/json?all=true&filters=%7B%22label%22%3A%5B%22com.docker.compose.project%3D${STACK_NAME}%22%5D%7D" \
|
"https://${PORTAINER_URL}/api/endpoints/${ENDPOINT_ID}/docker/containers/json?all=true&filters=%7B%22label%22%3A%5B%22com.docker.compose.project%3D${STACK_NAME}%22%5D%7D" \
|
||||||
-H "Authorization: Bearer ${TOKEN}")
|
-H "Authorization: Bearer ${TOKEN}")
|
||||||
|
|
||||||
TOTAL=$(echo "$CONTAINERS" | jq 'length')
|
# Container baru = ID tidak ada di daftar container lama
|
||||||
RUNNING=$(echo "$CONTAINERS" | jq '[.[] | select(.State == "running")] | length')
|
NEW_RUNNING=$(echo "$CONTAINERS" | jq \
|
||||||
FAILED=$(echo "$CONTAINERS" | jq '[.[] | select(.State == "exited" and (.Status | test("Exited \\(0\\)") | not))] | length')
|
--arg old "$OLD_IDS" \
|
||||||
|
'[.[] | select(.State == "running" and ((.Id) as $id | ($old | split(",") | index($id)) == null))] | length')
|
||||||
|
|
||||||
echo "🔄 [${COUNT}/${MAX_RETRY}] Running: ${RUNNING} | Failed: ${FAILED} | Total: ${TOTAL}"
|
FAILED=$(echo "$CONTAINERS" | jq \
|
||||||
echo "$CONTAINERS" | jq -r '.[] | " → \(.Names[0]) | \(.State) | \(.Status)"'
|
'[.[] | select(.State == "exited" and (.Status | test("Exited \\(0\\)") | not) and (.Names[0] | test("seed") | not))] | length')
|
||||||
|
|
||||||
|
echo "🔄 [$((COUNT * SLEEP_INTERVAL))s / $((MAX_RETRY * SLEEP_INTERVAL))s] Container baru running: ${NEW_RUNNING} | Gagal: ${FAILED}"
|
||||||
|
echo "$CONTAINERS" | jq -r '.[] | " → \(.Names[0]) | \(.State) | \(.Status) | id: \(.Id[:12])"'
|
||||||
|
|
||||||
if [ "$FAILED" -gt "0" ]; then
|
if [ "$FAILED" -gt "0" ]; then
|
||||||
echo ""
|
echo ""
|
||||||
echo "❌ Ada container yang crash!"
|
echo "❌ Ada container yang crash!"
|
||||||
echo "$CONTAINERS" | jq -r '.[] | select(.State == "exited" and (.Status | test("Exited \\(0\\)") | not)) | " → \(.Names[0]) | \(.Status)"'
|
echo "$CONTAINERS" | jq -r '.[] | select(.State == "exited" and (.Status | test("Exited \\(0\\)") | not) and (.Names[0] | test("seed") | not)) | " → \(.Names[0]) | \(.Status)"'
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [ "$RUNNING" -gt "0" ]; then
|
if [ "$NEW_RUNNING" -gt "0" ]; then
|
||||||
echo ""
|
echo ""
|
||||||
echo "✅ Stack $STACK_NAME berhasil di-redeploy dan running!"
|
echo "✅ Stack $STACK_NAME berhasil di-redeploy dengan image baru dan running!"
|
||||||
exit 0
|
exit 0
|
||||||
fi
|
fi
|
||||||
done
|
done
|
||||||
|
|
||||||
echo ""
|
echo ""
|
||||||
echo "❌ Timeout! Stack tidak kunjung running setelah $((MAX_RETRY * 5)) detik."
|
echo "❌ Timeout $((MAX_RETRY * SLEEP_INTERVAL))s! Container baru tidak kunjung running."
|
||||||
|
echo " Kemungkinan image masih dalam proses pull atau ada error di server."
|
||||||
exit 1
|
exit 1
|
||||||
Reference in New Issue
Block a user