Compare commits
30 Commits
fc6f97f4a2
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
| d4b92e8837 | |||
| 5bcfbe84a1 | |||
| 4c093b1696 | |||
| d67d0f936e | |||
| f93fb53638 | |||
| b4006cdeec | |||
| 38b5371b81 | |||
| 5a73ff5d93 | |||
| 8e643c59d8 | |||
| 9365725f30 | |||
| ee48dc8f82 | |||
| 5d61b61ed2 | |||
| c0baa5f8cd | |||
| 9142c30332 | |||
| b81692b835 | |||
| a5b73ee528 | |||
| 7ca2c9e568 | |||
| 663652e821 | |||
| b5fcd0e103 | |||
| 9a6d3d6551 | |||
| c8fc975303 | |||
| 6d6395c654 | |||
| 6378347ba6 | |||
| 86fd9cedf6 | |||
| 9566387db9 | |||
| cbb6a1bff7 | |||
| 6aed7c0492 | |||
| f1c2abb411 | |||
| 6635d8a0be | |||
| 295fbf3267 |
2
.gitignore
vendored
2
.gitignore
vendored
@@ -1 +1 @@
|
||||
config.conf.own
|
||||
config_own.conf
|
||||
|
||||
404
README.md
404
README.md
@@ -1,14 +1,26 @@
|
||||
# Docker Compose Auto-Updater
|
||||
|
||||
Dieses Script überprüft mehrere Docker-Compose-Stacks auf Image-Updates und aktualisiert diese automatisch.
|
||||
<p align="center">
|
||||
<img src="./images/composeupdater.png" alt="Logo" width="200">
|
||||
</p>
|
||||
|
||||
---
|
||||
|
||||
> 🔧 Automatisches Update von Docker-Compose-Stacks mit feingranularer Steuerung per Labels
|
||||
|
||||
## 🚀 Features
|
||||
|
||||
- 🔄 Stack-basiertes Update
|
||||
- 🧪 Dry-Run Modus
|
||||
- 📲 ntfy Benachrichtigungen
|
||||
- ⏭️ Exclude-Liste für ganze Stacks oder einzelne Container
|
||||
- 🗑️ Prune Funktion
|
||||
- 🔄 **Stack-basiertes Update**
|
||||
Aktualisiert komplette Docker-Compose Stacks strukturiert und kontrolliert
|
||||
- 🧪 **Dry-Run Modus**
|
||||
Zeigt an, was passieren würde, ohne Änderungen durchzuführen
|
||||
- 📲 **ntfy Benachrichtigungen**
|
||||
Push-Notifications über ntfy bei Updates, Fehlern oder Status
|
||||
- ⚙️ **Service-Modi** (per Label steuerbar)
|
||||
Einzelne Container oder komplette Stacks gezielt vom Update ausschließen
|
||||
- 🗑️ **Prune Funktion**
|
||||
Entfernt nicht mehr benötigte Images/Container automatisch
|
||||
|
||||
|
||||
---
|
||||
|
||||
@@ -16,122 +28,119 @@ Dieses Script überprüft mehrere Docker-Compose-Stacks auf Image-Updates und ak
|
||||
|
||||
- Docker + Docker Compose (v2)
|
||||
- Bash
|
||||
- jq
|
||||
- Optional: ntfy Server
|
||||
|
||||
---
|
||||
|
||||
## ⚙️ Konfiguration (`config.conf`)
|
||||
|
||||
```bash
|
||||
# =============================
|
||||
# =============================
|
||||
# Pfade
|
||||
# =============================
|
||||
|
||||
# Pfad zu deinen Compose-Files
|
||||
COMPOSE_DIR="/pfad/zu/deinen/stacks"
|
||||
# Logging
|
||||
LOG_FILE="/pfad/zum/log/update.log"
|
||||
LOG_LEVEL="INFO" # DEBUG=sehr detailliert, INFO=Standard, WARN=nur wichtige Hinweise/Updates, ERROR=nur Fehler
|
||||
|
||||
# Dateimuster
|
||||
COMPOSE_PATTERN="docker-compose.yml"
|
||||
|
||||
# =============================
|
||||
# =============================
|
||||
# Allgemein Einstellungen
|
||||
# =============================
|
||||
|
||||
# Verhalten bei gestoppten Containern
|
||||
UPDATE_STOPPED=true # Image aktualisieren
|
||||
START_STOPPED=false # danach NICHT starten
|
||||
|
||||
# Dry Run (true/false)
|
||||
DRY_RUN=false
|
||||
|
||||
# =============================
|
||||
# =============================
|
||||
# Exclude
|
||||
# =============================
|
||||
|
||||
# Exclude Container
|
||||
EXCLUDE_SERVICES=(
|
||||
"example_container_1"
|
||||
"example_container_2"
|
||||
)
|
||||
|
||||
# Exclude Stack
|
||||
EXCLUDE_STACKS=(
|
||||
"example_stack_1"
|
||||
"example_stack_2"
|
||||
)
|
||||
|
||||
# =============================
|
||||
# =============================
|
||||
# NTFY
|
||||
# =============================
|
||||
NTFY_ENABLED=true
|
||||
NTFY_TITLE="Docker Update ($(hostname))"
|
||||
NTFY_TOKEN="DEIN_TOKEN"
|
||||
NTFY_URL="https://ntfy.example.com/topic"
|
||||
NTFY_IMAGE_URL="http://dein-server/host-icon.png"
|
||||
NTFY_TAGS="docker,update"
|
||||
NTFY_ONLY_ON_CHANGES=false
|
||||
# Versions Nr. anzeigen (true/false)
|
||||
SHOW_VERSIONS=true
|
||||
|
||||
# =============================
|
||||
# =============================
|
||||
# Docker Cleanup
|
||||
# =============================
|
||||
|
||||
ENABLE_CLEANUP=true
|
||||
CLEANUP_ONLY_ON_UPDATE=true
|
||||
|
||||
# Images:
|
||||
# 🟢 dangling → docker image prune (nur <none> Images)
|
||||
# 🟡 unused → docker image prune -a (alle ungenutzten Images)
|
||||
CLEANUP_IMAGES=true
|
||||
CLEANUP_IMAGES_MODE="unused" # dangling | unused
|
||||
|
||||
# Container:
|
||||
# entfernt gestoppte Container
|
||||
# 🟢 docker container prune
|
||||
CLEANUP_CONTAINERS=true
|
||||
|
||||
# Volume:
|
||||
# entfernt ungenutzte Volumes
|
||||
# ⚠️ kann Daten löschen
|
||||
CLEANUP_VOLUMES=false
|
||||
|
||||
# Networks:
|
||||
# entfernt ungenutzte Netzwerke
|
||||
# 🟢 meist unkritisch
|
||||
CLEANUP_NETWORKS=true
|
||||
|
||||
# =============================
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## ▶️ Nutzung
|
||||
|
||||
```bash
|
||||
chmod +x script.sh
|
||||
./script.sh
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 🧠 Funktionsweise
|
||||
|
||||
1. Alle `docker-compose.yml` Dateien werden gefunden
|
||||
2. Alphabetisch sortiert
|
||||
3. Jeder Stack wird geprüft:
|
||||
- Image wird gepullt
|
||||
- Vergleich: Container Image-ID vs. aktuelles Image
|
||||
4. Wenn ein Service ein Update hat:
|
||||
- kompletter Stack wird neu deployed
|
||||
1. Alle `*compose*.yml` Dateien werden rekursiv gefunden
|
||||
2. Verarbeitung erfolgt alphabetisch (deterministische Reihenfolge)
|
||||
3. Für jeden Stack:
|
||||
- Compose-Konfiguration wird ausgewertet (`docker compose config`)
|
||||
- Services und deren Images werden ermittelt
|
||||
- Für jedes Image:
|
||||
- Image wird bei Bedarf gepullt (maximal einmal pro Image und Stack, Cache-basiert)
|
||||
- Lokale Image-ID wird ermittelt
|
||||
- Image-ID des vorhandenen Containers wird ermittelt (auch für gestoppte Container)
|
||||
4. Entscheidungslogik:
|
||||
- ❌ Container existiert nicht → kein Update (nur Definition vorhanden)
|
||||
- ❌ Image-ID unverändert → kein Update
|
||||
- ✅ Image-ID hat sich geändert → Update erkannt
|
||||
5. Wenn mindestens ein Service ein Update hat:
|
||||
- **Einzelcontainer:**
|
||||
- gezieltes Update nur dieses Services (`docker compose up -d <service>`)
|
||||
- **Mehrere Services:**
|
||||
- kompletter Stack wird neu deployed (`docker compose up -d`)
|
||||
- **Optional**
|
||||
- feste Wartezeit nach dem Deploy (`REDEPLOY_WAIT`)
|
||||
- anschließendes Warten auf erfolgreiche Healthchecks (`REDEPLOY_WAIT_HEALTHY`)
|
||||
6. Sonderverhalten:
|
||||
- Gestoppte Container werden ebenfalls geprüft und bei Updates berücksichtigt
|
||||
- Gestoppte Container werden nach dem Update optional wieder gestoppt
|
||||
- Service-Verhalten wird vollständig über Labels gesteuert (`composeupdater.mode`)
|
||||
- Je nach Modus werden Services:
|
||||
- komplett ignoriert (`ignore`)
|
||||
- nur überwacht (`notify-only`)
|
||||
- oder automatisch aktualisiert (`update`)
|
||||
|
||||
|
||||
---
|
||||
|
||||
## ⚙️ Service-Modi (per Label steuerbar)
|
||||
|
||||
Du kannst das Verhalten einzelner Services oder ganzer Stacks über Labels steuern:
|
||||
|
||||
```yml
|
||||
labels:
|
||||
- composeupdater.mode=update
|
||||
```
|
||||
|
||||
### 📊 Übersicht
|
||||
|
||||
| Mode | Pull | Compare | Update | ntfy |
|
||||
| ----------- | ---- | ------- | ------ | ---- |
|
||||
| update | ✅ | ✅ | ✅ | ✅ |
|
||||
| notify-only | ✅ | ✅ | ❌ | ✅ |
|
||||
| ignore | ❌ | ❌ | ❌ | ❌ |
|
||||
|
||||
|
||||
### 🧠 Erklärung der Modi
|
||||
|
||||
🔄 `update` (**Standard**)
|
||||
- Images werden gepullt
|
||||
- Lokales Image wird mit dem Container verglichen
|
||||
- Bei Änderungen wird der Service bzw. Stack aktualisiert
|
||||
- ntfy-Benachrichtigung wird gesendet
|
||||
|
||||
|
||||
🔔 `notify-only`
|
||||
- Image wird gepullt (für Vergleich notwendig)
|
||||
- Es wird geprüft, ob ein Update verfügbar ist
|
||||
- Kein Container-Update / kein Restart
|
||||
- ntfy informiert über verfügbare Updates
|
||||
|
||||
|
||||
🚫 `ignore`
|
||||
- Service wird komplett ignoriert
|
||||
- Kein Pull
|
||||
- Kein Vergleich
|
||||
- Kein Update
|
||||
- Keine Benachrichtigung
|
||||
|
||||
|
||||
### 🧩 Beispiele
|
||||
|
||||
#### Service ausschließen (komplett ignorieren)
|
||||
|
||||
```yml
|
||||
services:
|
||||
db:
|
||||
image: postgres:15
|
||||
labels:
|
||||
- composeupdater.mode=ignore
|
||||
```
|
||||
|
||||
#### Nur Benachrichtigung, kein automatisches Update
|
||||
|
||||
```yml
|
||||
services:
|
||||
app:
|
||||
image: myapp:latest
|
||||
labels:
|
||||
- composeupdater.mode=notify-only
|
||||
```
|
||||
|
||||
#### Explizit Standardverhalten setzen
|
||||
|
||||
```yml
|
||||
services:
|
||||
web:
|
||||
image: nginx:latest
|
||||
labels:
|
||||
- composeupdater.mode=update
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
@@ -157,7 +166,7 @@ Dabei werden ungenutzte Ressourcen entfernt:
|
||||
|
||||
---
|
||||
|
||||
## 🔔 ntfy Prioritäten
|
||||
## 🔔 ntfy
|
||||
|
||||
| Zustand | Priorität |
|
||||
|----------------------|----------|
|
||||
@@ -165,22 +174,163 @@ Dabei werden ungenutzte Ressourcen entfernt:
|
||||
| 🔄 Updates vorhanden | 3 |
|
||||
| ❌ Fehler | 5 |
|
||||
|
||||
### Anzeigebeispiel
|
||||
|
||||
<p align="left">
|
||||
<img src="./images/ntfy.png" alt="Logo" width="400">
|
||||
</p>
|
||||
|
||||
---
|
||||
|
||||
## ⚙️ Konfiguration (`config.conf`)
|
||||
|
||||
```bash
|
||||
# ==========================================================
|
||||
# DOCKER COMPOSE UPDATER - CONFIG
|
||||
# ==========================================================
|
||||
|
||||
|
||||
# ----------------------------------------------------------
|
||||
# PATH
|
||||
# ----------------------------------------------------------
|
||||
|
||||
# Basisverzeichnis der Stacks [ String ]
|
||||
PATH_COMPOSE_DIR="/pfad/zu/deinen/stacks"
|
||||
|
||||
# Compose-Dateiname [ String ]
|
||||
PATH_COMPOSE_PATTERN="*compose*.yml"
|
||||
|
||||
|
||||
# ----------------------------------------------------------
|
||||
# LOG
|
||||
# ----------------------------------------------------------
|
||||
|
||||
# Log-Datei [ String ]
|
||||
LOG_FILE="/pfad/zum/log/update.log"
|
||||
|
||||
# Log-Level [ DEBUG | INFO | WARN | ERROR ]
|
||||
LOG_LEVEL="INFO"
|
||||
|
||||
|
||||
# ----------------------------------------------------------
|
||||
# UPDATE
|
||||
# ----------------------------------------------------------
|
||||
|
||||
# Nur Simulation, keine Änderungen [ true | false ]
|
||||
UPDATE_DRY_RUN_ENABLED=false
|
||||
|
||||
# Gestoppte Container updaten [ true | false ]
|
||||
UPDATE_INCLUDE_STOPPED=true
|
||||
|
||||
# Danach wieder starten [ true | false ]
|
||||
UPDATE_START_STOPPED=false
|
||||
|
||||
|
||||
# ----------------------------------------------------------
|
||||
# REDEPLOY
|
||||
# ----------------------------------------------------------
|
||||
|
||||
# Feste Wartezeit nach Redeploy [ Number ]
|
||||
REDEPLOY_WAIT=45
|
||||
|
||||
# Warten bis Container healthy sind [ true | false ]
|
||||
REDEPLOY_WAIT_HEALTHY=true
|
||||
|
||||
# Timeout in Sekunden für healthy Check [ Number ]
|
||||
REDEPLOY_WAIT_HEALTHY_TIMEOUT=60
|
||||
|
||||
|
||||
# ----------------------------------------------------------
|
||||
# NTFY SETTINGS
|
||||
# ----------------------------------------------------------
|
||||
|
||||
# NTFY Zusammenfassung senden [ true | false ]
|
||||
NTFY_ENABLED=true
|
||||
|
||||
# Server URL [ String ]
|
||||
NTFY_URL="https://ntfy.example.com/topic"
|
||||
|
||||
# Token [ String ]
|
||||
NTFY_TOKEN="DEIN_TOKEN"
|
||||
|
||||
# Titel mitsenden (optional) [ String ]
|
||||
NTFY_TITLE="Autoupdate Report ($(hostname))"
|
||||
|
||||
# Tags mitsenden (optional) [ String ]
|
||||
NTFY_TAGS="docker,update"
|
||||
|
||||
# Icon mitsenden (optional) [ String ]
|
||||
NTFY_IMAGE_URL="http://dein-server/host-icon.png"
|
||||
|
||||
# Nur senden wenn Updates vorhanden [ true | false ]
|
||||
NTFY_ONLY_ON_CHANGES=false
|
||||
|
||||
# Versionsnummern anzeigen [ true | false ]
|
||||
NTFY_SHOW_VERSIONS=true
|
||||
|
||||
|
||||
# ----------------------------------------------------------
|
||||
# DOCKER CLEANUP
|
||||
# ----------------------------------------------------------
|
||||
|
||||
# Prune Befehle ausführen [ true | false ]
|
||||
CLEANUP_ENABLED=true
|
||||
|
||||
# Nur nach erfolgten Updates ausführen [ true | false ]
|
||||
CLEANUP_ONLY_ON_UPDATE=true
|
||||
|
||||
# Images löschen [ true | false ]
|
||||
CLEANUP_IMAGES_ENABLED=true
|
||||
|
||||
# Image-Prune Modus [ dangling | unused ]
|
||||
CLEANUP_IMAGES_MODE="unused"
|
||||
|
||||
# Container löschen [ true | false ]
|
||||
CLEANUP_CONTAINERS_ENABLED=true
|
||||
|
||||
# Volumes löschen [ true | false ]
|
||||
CLEANUP_VOLUMES_ENABLED=false
|
||||
|
||||
# Networks löschen [ true | false ]
|
||||
CLEANUP_NETWORKS_ENABLED=true
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## ▶️ Nutzung
|
||||
|
||||
```bash
|
||||
chmod +x script.sh
|
||||
./script.sh
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
## 📄 Beispiel Ausgabe
|
||||
|
||||
```
|
||||
→ Prüfe Stack: homepage
|
||||
├─ dockerproxy (image)
|
||||
└─ homepage (image)
|
||||
|
||||
→ Prüfe Stack: app
|
||||
├─ db (image)
|
||||
🔍 Prüfe Stack: rss
|
||||
├─ read (phpdockerio/readability-js-server) [Mode: 🔄 update]
|
||||
├─ merc (wangqiru/mercury-parser-api) [Mode: 🔄 update]
|
||||
├─ full-text-rss (heussd/fivefilters-full-text-rss:latest) [Mode: 🔄 update]
|
||||
├─ rss-bridge (rssbridge/rss-bridge:latest) [Mode: 🔄 update]
|
||||
⬆️ UPDATE
|
||||
alt: sha256:abc
|
||||
neu: sha256:def
|
||||
└─ web (image)
|
||||
🔄 Stack wird neu deployt
|
||||
alt: rssbridge/rss-bridge:latest@sha256:55215923cf81b2fa6fbb7ecc1bd2555405f4fc06029ae9876e91164a735c7b9d
|
||||
neu: rssbridge/rss-bridge:latest@sha256:f3f0218c8b075cbc7c559c8e6852888e95fa6d68258436da6195efc5ab98b025
|
||||
└─ freshrss (freshrss/freshrss:latest) [Mode: 🔄 update]
|
||||
♻️ Stack wird neu deployt (Trigger: rss-bridge)
|
||||
⏳ Deploy läuft...
|
||||
✅ Stack erfolgreich aktualisiert
|
||||
💤 Warte 60s nach Deploy
|
||||
⏳ Warte auf healthy Container (max 60s)
|
||||
💚 Alle Container healthy
|
||||
|
||||
🕒 Dauer: 78s
|
||||
|
||||
|
||||
→ Prüfe Stack: tinymediamanager
|
||||
└─ tinymediamanager [Mode: 🚫 ignore]
|
||||
🕒 Dauer: 1s
|
||||
```
|
||||
|
||||
---
|
||||
|
||||
151
config.conf
151
config.conf
@@ -1,87 +1,108 @@
|
||||
# =============================
|
||||
# =============================
|
||||
# Pfade
|
||||
# =============================
|
||||
# ==========================================================
|
||||
# DOCKER COMPOSE UPDATER - CONFIG
|
||||
# ==========================================================
|
||||
|
||||
# Pfad zu deinen Compose-Files
|
||||
COMPOSE_DIR="/pfad/zu/deinen/stacks"
|
||||
# Logging
|
||||
|
||||
# ----------------------------------------------------------
|
||||
# PATH
|
||||
# ----------------------------------------------------------
|
||||
|
||||
# Basisverzeichnis der Stacks [ String ]
|
||||
PATH_COMPOSE_DIR="/pfad/zu/deinen/stacks"
|
||||
|
||||
# Compose-Dateiname [ String ]
|
||||
PATH_COMPOSE_PATTERN="*compose*.yml"
|
||||
|
||||
|
||||
# ----------------------------------------------------------
|
||||
# LOG
|
||||
# ----------------------------------------------------------
|
||||
|
||||
# Log-Datei [ String ]
|
||||
LOG_FILE="/pfad/zum/log/update.log"
|
||||
LOG_LEVEL="INFO" # DEBUG=sehr detailliert, INFO=Standard, WARN=nur wichtige Hinweise/Updates, ERROR=nur Fehler
|
||||
|
||||
# Dateimuster
|
||||
COMPOSE_PATTERN="docker-compose.yml"
|
||||
# Log-Level [ DEBUG | INFO | WARN | ERROR ]
|
||||
LOG_LEVEL="INFO"
|
||||
|
||||
# =============================
|
||||
# =============================
|
||||
# Allgemein Einstellungen
|
||||
# =============================
|
||||
|
||||
# Verhalten bei gestoppten Containern
|
||||
UPDATE_STOPPED=true # Image aktualisieren
|
||||
START_STOPPED=false # danach NICHT starten
|
||||
# ----------------------------------------------------------
|
||||
# UPDATE
|
||||
# ----------------------------------------------------------
|
||||
|
||||
# Dry Run (true/false)
|
||||
DRY_RUN=false
|
||||
# Nur Simulation, keine Änderungen [ true | false ]
|
||||
UPDATE_DRY_RUN_ENABLED=false
|
||||
|
||||
# =============================
|
||||
# =============================
|
||||
# Exclude
|
||||
# =============================
|
||||
# Gestoppte Container updaten [ true | false ]
|
||||
UPDATE_INCLUDE_STOPPED=true
|
||||
|
||||
# Exclude Container
|
||||
EXCLUDE_SERVICES=(
|
||||
"example_container_1"
|
||||
"example_container_2"
|
||||
)
|
||||
# Danach wieder starten [ true | false ]
|
||||
UPDATE_START_STOPPED=false
|
||||
|
||||
# Exclude Stack
|
||||
EXCLUDE_STACKS=(
|
||||
"example_stack_1"
|
||||
"example_stack_2"
|
||||
)
|
||||
|
||||
# =============================
|
||||
# =============================
|
||||
# NTFY
|
||||
# =============================
|
||||
# ----------------------------------------------------------
|
||||
# REDEPLOY
|
||||
# ----------------------------------------------------------
|
||||
|
||||
# Feste Wartezeit nach Redeploy [ Number ]
|
||||
REDEPLOY_WAIT=120
|
||||
|
||||
# Warten bis Container healthy sind [ true | false ]
|
||||
REDEPLOY_WAIT_HEALTHY=true
|
||||
|
||||
# Timeout in Sekunden für healthy Check [ Number ]
|
||||
REDEPLOY_WAIT_HEALTHY_TIMEOUT=60
|
||||
|
||||
|
||||
# ----------------------------------------------------------
|
||||
# NTFY SETTINGS
|
||||
# ----------------------------------------------------------
|
||||
|
||||
# NTFY Zusammenfassung senden [ true | false ]
|
||||
NTFY_ENABLED=true
|
||||
NTFY_TITLE="Docker Update ($(hostname))"
|
||||
NTFY_TOKEN="DEIN_TOKEN"
|
||||
|
||||
# Server URL [ String ]
|
||||
NTFY_URL="https://ntfy.example.com/topic"
|
||||
NTFY_IMAGE_URL="http://dein-server/host-icon.png"
|
||||
|
||||
# Token [ String ]
|
||||
NTFY_TOKEN="DEIN_TOKEN"
|
||||
|
||||
# Titel mitsenden (optional) [ String ]
|
||||
NTFY_TITLE="Autoupdate Report ($(hostname))"
|
||||
|
||||
# Tags mitsenden (optional) [ String ]
|
||||
NTFY_TAGS="docker,update"
|
||||
|
||||
# Icon mitsenden (optional) [ String ]
|
||||
NTFY_IMAGE_URL="http://dein-server/host-icon.png"
|
||||
|
||||
# Nur senden wenn Updates vorhanden [ true | false ]
|
||||
NTFY_ONLY_ON_CHANGES=false
|
||||
# Versions Nr. anzeigen (true/false)
|
||||
SHOW_VERSIONS=true
|
||||
|
||||
# =============================
|
||||
# =============================
|
||||
# Docker Cleanup
|
||||
# =============================
|
||||
# Versionsnummern anzeigen [ true | false ]
|
||||
NTFY_SHOW_VERSIONS=true
|
||||
|
||||
ENABLE_CLEANUP=true
|
||||
|
||||
# ----------------------------------------------------------
|
||||
# DOCKER CLEANUP
|
||||
# ----------------------------------------------------------
|
||||
|
||||
# Prune Befehle ausführen [ true | false ]
|
||||
CLEANUP_ENABLED=true
|
||||
|
||||
# Nur nach erfolgten Updates ausführen [ true | false ]
|
||||
CLEANUP_ONLY_ON_UPDATE=true
|
||||
|
||||
# Images:
|
||||
# 🟢 dangling → docker image prune (nur <none> Images)
|
||||
# 🟢 unused → docker image prune -a (alle ungenutzten Images)
|
||||
CLEANUP_IMAGES=true
|
||||
CLEANUP_IMAGES_MODE="unused" # dangling | unused
|
||||
# Images löschen [ true | false ]
|
||||
CLEANUP_IMAGES_ENABLED=true
|
||||
|
||||
# Container:
|
||||
# entfernt gestoppte Container
|
||||
# 🟢 docker container prune
|
||||
CLEANUP_CONTAINERS=true
|
||||
# Image-Prune Modus [ dangling | unused ]
|
||||
CLEANUP_IMAGES_MODE="unused"
|
||||
|
||||
# Volume:
|
||||
# entfernt ungenutzte Volumes
|
||||
# ⚠️ kann Daten löschen
|
||||
CLEANUP_VOLUMES=false
|
||||
# Container löschen [ true | false ]
|
||||
CLEANUP_CONTAINERS_ENABLED=true
|
||||
|
||||
# Networks:
|
||||
# entfernt ungenutzte Netzwerke
|
||||
# 🟢 meist unkritisch
|
||||
CLEANUP_NETWORKS=true
|
||||
# Volumes löschen [ true | false ]
|
||||
CLEANUP_VOLUMES_ENABLED=false
|
||||
|
||||
# =============================
|
||||
# Networks löschen [ true | false ]
|
||||
CLEANUP_NETWORKS_ENABLED=true
|
||||
87
config.conf_
87
config.conf_
@@ -1,87 +0,0 @@
|
||||
# =============================
|
||||
# =============================
|
||||
# Pfade
|
||||
# =============================
|
||||
|
||||
# Pfad zu deinen Compose-Files
|
||||
COMPOSE_DIR="/pfad/zu/deinen/stacks"
|
||||
# Logging
|
||||
LOG_FILE="/pfad/zum/log/update.log"
|
||||
LOG_LEVEL="INFO" # DEBUG=sehr detailliert, INFO=Standard, WARN=nur wichtige Hinweise/Updates, ERROR=nur Fehler
|
||||
|
||||
# Dateimuster
|
||||
COMPOSE_PATTERN="docker-compose.yml"
|
||||
|
||||
# =============================
|
||||
# =============================
|
||||
# Allgemein Einstellungen
|
||||
# =============================
|
||||
|
||||
# Verhalten bei gestoppten Containern
|
||||
UPDATE_STOPPED=true # Image aktualisieren
|
||||
START_STOPPED=false # danach NICHT starten
|
||||
|
||||
# Dry Run (true/false)
|
||||
DRY_RUN=false
|
||||
|
||||
# =============================
|
||||
# =============================
|
||||
# Exclude
|
||||
# =============================
|
||||
|
||||
# Exclude Container
|
||||
EXCLUDE_SERVICES=(
|
||||
"example_container_1"
|
||||
"example_container_2"
|
||||
)
|
||||
|
||||
# Exclude Stack
|
||||
EXCLUDE_STACKS=(
|
||||
"example_stack_1"
|
||||
"example_stack_2"
|
||||
)
|
||||
|
||||
# =============================
|
||||
# =============================
|
||||
# NTFY
|
||||
# =============================
|
||||
NTFY_ENABLED=true
|
||||
NTFY_TITLE="Docker Update ($(hostname))"
|
||||
NTFY_TOKEN="DEIN_TOKEN"
|
||||
NTFY_URL="https://ntfy.example.com/topic"
|
||||
NTFY_IMAGE_URL="http://dein-server/composeupdater.png"
|
||||
NTFY_TAGS="docker,update"
|
||||
NTFY_ONLY_ON_CHANGES=false
|
||||
# Versions Nr. anzeigen (true/false)
|
||||
SHOW_VERSIONS=true
|
||||
|
||||
# =============================
|
||||
# =============================
|
||||
# Docker Cleanup
|
||||
# =============================
|
||||
|
||||
ENABLE_CLEANUP=true
|
||||
CLEANUP_ONLY_ON_UPDATE=true
|
||||
|
||||
# Images:
|
||||
# 🟢 dangling → docker image prune (nur <none> Images)
|
||||
# 🟢 unused → docker image prune -a (alle ungenutzten Images)
|
||||
CLEANUP_IMAGES=true
|
||||
CLEANUP_IMAGES_MODE="unused" # dangling | unused
|
||||
|
||||
# Container:
|
||||
# entfernt gestoppte Container
|
||||
# 🟢 docker container prune
|
||||
CLEANUP_CONTAINERS=true
|
||||
|
||||
# Volume:
|
||||
# entfernt ungenutzte Volumes
|
||||
# ⚠️ kann Daten löschen
|
||||
CLEANUP_VOLUMES=false
|
||||
|
||||
# Networks:
|
||||
# entfernt ungenutzte Netzwerke
|
||||
# 🟢 meist unkritisch
|
||||
CLEANUP_NETWORKS=true
|
||||
|
||||
# =============================
|
||||
|
Before Width: | Height: | Size: 266 KiB After Width: | Height: | Size: 266 KiB |
BIN
images/ntfy.png
Normal file
BIN
images/ntfy.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 163 KiB |
@@ -1,5 +1,4 @@
|
||||
#!/bin/bash
|
||||
|
||||
set -euo pipefail
|
||||
|
||||
BASE_DIR="$(cd "$(dirname "$0")" && pwd)"
|
||||
@@ -19,24 +18,22 @@ level_to_int() {
|
||||
esac
|
||||
}
|
||||
|
||||
INDENT=" "
|
||||
SUBINDENT=" "
|
||||
|
||||
should_log() {
|
||||
[ "$(level_to_int "$1")" -ge "$(level_to_int "$LOG_LEVEL")" ]
|
||||
}
|
||||
|
||||
log() {
|
||||
local level="$1"
|
||||
shift
|
||||
local level="$1"; shift
|
||||
local msg="$*"
|
||||
|
||||
if should_log "$level"; then
|
||||
echo "$(date '+%Y-%m-%d %H:%M:%S') | $level | $msg" | tee -a "$LOG_FILE"
|
||||
fi
|
||||
}
|
||||
|
||||
# =============================
|
||||
# Log begrenzen
|
||||
# =============================
|
||||
|
||||
if [ -f "$LOG_FILE" ]; then
|
||||
line_count=$(wc -l < "$LOG_FILE")
|
||||
if [ "$line_count" -gt 1000 ]; then
|
||||
@@ -48,34 +45,34 @@ fi
|
||||
# Helper
|
||||
# =============================
|
||||
|
||||
is_excluded() {
|
||||
|
||||
get_service_mode() {
|
||||
local svc="$1"
|
||||
for ex in "${EXCLUDE_SERVICES[@]}"; do
|
||||
[[ "$svc" == "$ex" ]] && return 0
|
||||
done
|
||||
return 1
|
||||
|
||||
echo "$compose_json" \
|
||||
| jq -r --arg svc "$svc" '
|
||||
.services[$svc].labels // {}
|
||||
| (if type=="array"
|
||||
then map(split("=") | {(.[0]): .[1]}) | add
|
||||
else .
|
||||
end)
|
||||
| .["composeupdater.mode"] // "update"
|
||||
'
|
||||
}
|
||||
|
||||
is_stack_excluded() {
|
||||
local stack="$1"
|
||||
for ex in "${EXCLUDE_STACKS[@]}"; do
|
||||
[[ "$stack" == "$ex" ]] && return 0
|
||||
done
|
||||
return 1
|
||||
}
|
||||
|
||||
get_image() {
|
||||
local svc="$1"
|
||||
docker compose config | awk "/^ $svc:/,/image:/" | grep image | head -n1 | awk '{print $2}'
|
||||
|
||||
echo "$compose_json" \
|
||||
| jq -r --arg svc "$svc" '.services[$svc].image // empty'
|
||||
}
|
||||
|
||||
get_container_image_id() {
|
||||
local svc="$1"
|
||||
local cid
|
||||
|
||||
cid=$(docker compose ps -q "$svc" 2>/dev/null || true)
|
||||
cid=$(docker compose ps -aq "$svc" | head -n1)
|
||||
[ -z "$cid" ] && echo "" && return
|
||||
|
||||
docker inspect -f '{{.Image}}' "$cid" 2>/dev/null || echo ""
|
||||
}
|
||||
|
||||
@@ -84,33 +81,16 @@ get_local_image_id() {
|
||||
docker image inspect -f '{{.Id}}' "$image" 2>/dev/null || echo ""
|
||||
}
|
||||
|
||||
get_container_image_ref() {
|
||||
local svc="$1"
|
||||
local cid
|
||||
|
||||
cid=$(docker compose ps -q "$svc" 2>/dev/null || true)
|
||||
[ -z "$cid" ] && echo "none" && return
|
||||
|
||||
docker inspect -f '{{.Config.Image}}' "$cid" 2>/dev/null || echo "unknown"
|
||||
}
|
||||
|
||||
get_local_image_digest() {
|
||||
local image="$1"
|
||||
docker inspect --format='{{index .RepoDigests 0}}' "$image" 2>/dev/null || echo "unknown"
|
||||
}
|
||||
|
||||
is_running() {
|
||||
local svc="$1"
|
||||
local cid
|
||||
|
||||
cid=$(docker compose ps -q "$svc" 2>/dev/null || true)
|
||||
cid=$(docker compose ps -aq "$svc" 2>/dev/null || true)
|
||||
[ -z "$cid" ] && return 1
|
||||
|
||||
docker inspect -f '{{.State.Running}}' "$cid" 2>/dev/null | grep -q true
|
||||
}
|
||||
|
||||
run_cmd() {
|
||||
if [ "$DRY_RUN" = true ]; then
|
||||
if [ "$UPDATE_DRY_RUN_ENABLED" = true ]; then
|
||||
log DEBUG "[DRY RUN] $*"
|
||||
else
|
||||
eval "$@"
|
||||
@@ -122,33 +102,22 @@ send_ntfy() {
|
||||
local prio="$2"
|
||||
local image_url="${NTFY_IMAGE_URL:-}"
|
||||
|
||||
if [ -n "$image_url" ]; then
|
||||
curl -s \
|
||||
-H "Authorization: Bearer $NTFY_TOKEN" \
|
||||
-H "Title: $NTFY_TITLE" \
|
||||
-H "Priority: $prio" \
|
||||
-H "Tags: $NTFY_TAGS" \
|
||||
-H "Icon: $image_url" \
|
||||
-H "Markdown: yes" \
|
||||
${image_url:+-H "Icon: $image_url"} \
|
||||
-d "$msg" \
|
||||
"$NTFY_URL" > /dev/null || true
|
||||
else
|
||||
curl -s \
|
||||
-H "Authorization: Bearer $NTFY_TOKEN" \
|
||||
-H "Title: $NTFY_TITLE" \
|
||||
-H "Priority: $prio" \
|
||||
-H "Tags: $NTFY_TAGS" \
|
||||
-d "$msg" \
|
||||
"$NTFY_URL" > /dev/null || true
|
||||
fi
|
||||
}
|
||||
|
||||
get_docker_disk_usage() {
|
||||
local total=0
|
||||
|
||||
while read -r size; do
|
||||
num="${size//[!0-9.]/}"
|
||||
num="${num:-0}"
|
||||
|
||||
if [[ "$size" == *GB ]]; then
|
||||
total=$(bc <<< "$total + ($num * 1024)")
|
||||
elif [[ "$size" == *MB ]]; then
|
||||
@@ -161,54 +130,153 @@ get_docker_disk_usage() {
|
||||
LC_NUMERIC=C printf "%.0f\n" "$total"
|
||||
}
|
||||
|
||||
wait_for_healthy() {
|
||||
local timeout="$1"
|
||||
shift
|
||||
local services=("$@")
|
||||
|
||||
local start
|
||||
start=$(date +%s)
|
||||
|
||||
# Containerliste bestimmen
|
||||
if [ ${#services[@]} -gt 0 ]; then
|
||||
mapfile -t cids < <(docker compose ps -aq "${services[@]}")
|
||||
else
|
||||
mapfile -t cids < <(docker compose ps -aq)
|
||||
fi
|
||||
|
||||
# keine Container → nichts zu tun
|
||||
if [ ${#cids[@]} -eq 0 ]; then
|
||||
log INFO "${SUBINDENT}ℹ️ Keine Container gefunden → überspringe Healthcheck"
|
||||
return 0
|
||||
fi
|
||||
|
||||
# prüfen ob überhaupt Healthchecks existieren
|
||||
local has_healthcheck=false
|
||||
|
||||
for cid in "${cids[@]}"; do
|
||||
health=$(docker inspect -f '{{if .State.Health}}{{.State.Health.Status}}{{end}}' "$cid" 2>/dev/null)
|
||||
|
||||
if [ -n "$health" ]; then
|
||||
has_healthcheck=true
|
||||
break
|
||||
fi
|
||||
done
|
||||
|
||||
# wenn keiner vorhanden → direkt raus
|
||||
if [ "$has_healthcheck" = false ]; then
|
||||
log INFO "${SUBINDENT}ℹ️ Keine Healthchecks → überspringe warten"
|
||||
return 0
|
||||
fi
|
||||
|
||||
log INFO "${SUBINDENT}⏳ Warte auf healthy Container (max ${timeout}s)"
|
||||
|
||||
while true; do
|
||||
local all_ok=true
|
||||
|
||||
for cid in "${cids[@]}"; do
|
||||
status=$(docker inspect -f '{{.State.Health.Status}}' "$cid" 2>/dev/null || echo "none")
|
||||
|
||||
if [ "$status" = "starting" ] || [ "$status" = "unhealthy" ]; then
|
||||
all_ok=false
|
||||
break
|
||||
fi
|
||||
done
|
||||
|
||||
if [ "$all_ok" = true ]; then
|
||||
log INFO "${SUBINDENT}💚 Alle Container healthy"
|
||||
return 0
|
||||
fi
|
||||
|
||||
local now
|
||||
now=$(date +%s)
|
||||
|
||||
if [ $((now - start)) -ge "$timeout" ]; then
|
||||
log WARN "${SUBINDENT}⚠️ Healthcheck Timeout erreicht"
|
||||
return 1
|
||||
fi
|
||||
|
||||
sleep 2
|
||||
done
|
||||
}
|
||||
|
||||
pull_with_retry() {
|
||||
local image="$1"
|
||||
|
||||
local retries=3
|
||||
local delay=3
|
||||
|
||||
for ((i=1; i<=retries; i++)); do
|
||||
|
||||
if docker pull "$image" >/dev/null 2>&1; then
|
||||
return 0
|
||||
fi
|
||||
|
||||
if [ "$i" -lt "$retries" ]; then
|
||||
log WARN " ⚠️ Pull fehlgeschlagen – Versuch $i/$retries, retry in ${delay}s"
|
||||
sleep "$delay"
|
||||
fi
|
||||
done
|
||||
|
||||
return 1
|
||||
}
|
||||
|
||||
container_exists() {
|
||||
local svc="$1"
|
||||
docker compose ps -aq "$svc" 2>/dev/null | grep -q .
|
||||
}
|
||||
|
||||
|
||||
# =============================
|
||||
# Start
|
||||
# =============================
|
||||
|
||||
log INFO "==== Docker Compose Update gestartet ===="
|
||||
|
||||
script_start=$(date +%s)
|
||||
|
||||
notify_stacks_updated=()
|
||||
notify_excluded_updates=()
|
||||
error_flag=false
|
||||
|
||||
cd "$COMPOSE_DIR"
|
||||
declare -A stack_tree
|
||||
stack_tree=()
|
||||
|
||||
cd "$PATH_COMPOSE_DIR"
|
||||
|
||||
while IFS= read -r -d '' file; do
|
||||
|
||||
declare -A pulled_images
|
||||
|
||||
dir=$(dirname "$file")
|
||||
stack=$(basename "$dir")
|
||||
|
||||
if is_stack_excluded "$stack"; then
|
||||
log INFO "→ Stack $stack übersprungen (excluded)"
|
||||
continue
|
||||
fi
|
||||
|
||||
log INFO ""
|
||||
log INFO "→ Prüfe Stack: $stack"
|
||||
log INFO "🔍 Prüfe Stack: $stack"
|
||||
|
||||
stack_start=$(date +%s)
|
||||
|
||||
cd "$dir"
|
||||
|
||||
mapfile -t services < <(docker compose config --services)
|
||||
compose_json=$(docker compose config --format json)
|
||||
|
||||
mapfile -t services < <(docker compose config --services)
|
||||
total_services=${#services[@]}
|
||||
current_index=0
|
||||
|
||||
# Running State merken
|
||||
declare -A was_running
|
||||
for svc in "${services[@]}"; do
|
||||
if is_running "$svc"; then
|
||||
was_running["$svc"]=1
|
||||
else
|
||||
was_running["$svc"]=0
|
||||
fi
|
||||
is_running "$svc" && was_running[$svc]=1 || was_running[$svc]=0
|
||||
done
|
||||
|
||||
stack_updated=false
|
||||
changed_services=()
|
||||
version_report=()
|
||||
stack_block=""
|
||||
update_lines=()
|
||||
|
||||
for svc in "${services[@]}"; do
|
||||
|
||||
update_needed=false
|
||||
current_index=$((current_index + 1))
|
||||
|
||||
if [ "$current_index" -eq "$total_services" ]; then
|
||||
@@ -224,39 +292,110 @@ while IFS= read -r -d '' file; do
|
||||
continue
|
||||
fi
|
||||
|
||||
if is_excluded "$svc"; then
|
||||
docker pull "$image" >/dev/null 2>&1 || true
|
||||
notify_excluded_updates+=("$stack/$svc")
|
||||
log INFO " $prefix $svc (excluded)"
|
||||
# =============================
|
||||
# Mode (nur Service!)
|
||||
# =============================
|
||||
|
||||
mode=$(get_service_mode "$svc")
|
||||
|
||||
case "$mode" in
|
||||
update) mode_label="Mode: 🔄 update" ;;
|
||||
notify-only) mode_label="Mode: 🔔 notify-only" ;;
|
||||
ignore) mode_label="Mode: 🚫 ignore" ;;
|
||||
*)
|
||||
mode_label="Mode: ❓ unknown"
|
||||
mode="update"
|
||||
;;
|
||||
esac
|
||||
|
||||
case "$mode" in
|
||||
ignore)
|
||||
log INFO "${INDENT}$prefix $svc [$mode_label]"
|
||||
continue
|
||||
;;
|
||||
|
||||
notify-only)
|
||||
log INFO "${INDENT}$prefix $svc ($image) [$mode_label]"
|
||||
|
||||
before_id=$(get_container_image_id "$svc")
|
||||
|
||||
if [ -z "$before_id" ]; then
|
||||
continue
|
||||
fi
|
||||
|
||||
log INFO " $prefix $svc ($image)"
|
||||
|
||||
before_id=$(get_container_image_id "$svc")
|
||||
before_digest=$(docker inspect -f '{{.Image}}' "$(docker compose ps -q "$svc")" 2>/dev/null || echo "none")
|
||||
|
||||
if ! docker pull "$image" >/dev/null 2>&1; then
|
||||
log ERROR " $prefix ❌ Pull fehlgeschlagen"
|
||||
error_flag=true
|
||||
continue
|
||||
if [ -z "${pulled_images[$image]:-}" ]; then
|
||||
pull_with_retry "$image" || true
|
||||
pulled_images[$image]=1
|
||||
fi
|
||||
|
||||
after_id=$(get_local_image_id "$image")
|
||||
after_digest=$(get_local_image_digest "$image")
|
||||
|
||||
if [ -n "$before_id" ] && [ "$before_id" != "$after_id" ]; then
|
||||
if [ "$before_id" != "$after_id" ]; then
|
||||
notify_excluded_updates+=("$stack/$svc")
|
||||
fi
|
||||
|
||||
continue
|
||||
;;
|
||||
|
||||
update)
|
||||
log INFO "${INDENT}$prefix $svc ($image) [$mode_label]"
|
||||
;;
|
||||
esac
|
||||
|
||||
|
||||
before_id=$(get_container_image_id "$svc")
|
||||
|
||||
if [ -z "$before_id" ] && [ "$UPDATE_INCLUDE_STOPPED" = false ]; then
|
||||
log INFO "${SUBINDENT}⏭️ übersprungen (kein Container vorhanden)"
|
||||
continue
|
||||
fi
|
||||
|
||||
# =============================
|
||||
# Pull + Vergleich
|
||||
# =============================
|
||||
|
||||
if [ -z "${pulled_images[$image]:-}" ]; then
|
||||
if ! pull_with_retry "$image"; then
|
||||
log ERROR "${INDENT}❌ Pull fehlgeschlagen"
|
||||
error_flag=true
|
||||
continue
|
||||
fi
|
||||
pulled_images[$image]=1
|
||||
else
|
||||
log DEBUG "${SUBINDENT}⏩ Pull übersprungen (bereits gemacht)"
|
||||
fi
|
||||
|
||||
after_id=$(get_local_image_id "$image")
|
||||
|
||||
if ! container_exists "$svc"; then
|
||||
update_needed=false
|
||||
|
||||
elif [ "$before_id" != "$after_id" ]; then
|
||||
update_needed=true
|
||||
|
||||
else
|
||||
update_needed=false
|
||||
fi
|
||||
|
||||
# =============================
|
||||
# Update erkannt
|
||||
# =============================
|
||||
|
||||
if [ "$update_needed" = true ]; then
|
||||
|
||||
stack_updated=true
|
||||
changed_services+=("$svc")
|
||||
|
||||
if [ "$SHOW_VERSIONS" = true ]; then
|
||||
log INFO " ⬆️ UPDATE"
|
||||
log INFO " alt: $before_digest"
|
||||
log INFO " neu: $after_digest"
|
||||
version_report+=("$svc: ${after_digest##*@} → ${after_digest##*@}")
|
||||
else
|
||||
log INFO " ⬆️ UPDATE"
|
||||
fi
|
||||
log INFO "${SUBINDENT}⬆️ UPDATE"
|
||||
log INFO "${SUBINDENT} alt: ${image}@${before_id}"
|
||||
log INFO "${SUBINDENT} neu: ${image}@${after_id}"
|
||||
|
||||
short_before="${before_id#sha256:}"
|
||||
short_before="${short_before:0:6}"
|
||||
short_after="${after_id#sha256:}"
|
||||
short_after="${short_after:0:6}"
|
||||
|
||||
update_lines+=("$svc|$short_before|$short_after")
|
||||
fi
|
||||
done
|
||||
|
||||
@@ -269,64 +408,108 @@ while IFS= read -r -d '' file; do
|
||||
if [ "$total_services" -eq 1 ]; then
|
||||
svc="${services[0]}"
|
||||
|
||||
log INFO " 🔄 Einzelcontainer-Update: $svc"
|
||||
log INFO ""
|
||||
log INFO "${INDENT}🔄 Einzelcontainer-Update: $svc"
|
||||
|
||||
if [ "${was_running[$svc]}" = 1 ]; then
|
||||
if ! run_cmd docker compose up -d "$svc" --remove-orphans --no-color >/dev/null 2>&1; then
|
||||
log ERROR " ❌ Update fehlgeschlagen für $svc"
|
||||
error_flag=true
|
||||
run_cmd docker compose up -d "$svc" --remove-orphans >/dev/null 2>&1
|
||||
else
|
||||
log INFO " ✔️ Container $svc aktualisiert"
|
||||
fi
|
||||
if [ "$UPDATE_START_STOPPED" = true ]; then
|
||||
run_cmd docker compose up -d "$svc" --remove-orphans >/dev/null 2>&1
|
||||
else
|
||||
if ! run_cmd docker compose create "$svc" >/dev/null 2>&1; then
|
||||
log ERROR " ❌ Create fehlgeschlagen für $svc"
|
||||
error_flag=true
|
||||
else
|
||||
log INFO " ✔️ Container $svc aktualisiert (gestoppt)"
|
||||
run_cmd docker compose create "$svc"
|
||||
fi
|
||||
fi
|
||||
|
||||
notify_stacks_updated+=("$stack ($svc)")
|
||||
log INFO "${SUBINDENT}✅ Container aktualisiert"
|
||||
|
||||
if [ "${REDEPLOY_WAIT:-0}" -gt 0 ]; then
|
||||
log INFO "${SUBINDENT}💤 Warte ${REDEPLOY_WAIT}s nach Deploy"
|
||||
sleep "$REDEPLOY_WAIT"
|
||||
fi
|
||||
|
||||
if [ "$REDEPLOY_WAIT_HEALTHY" = true ]; then
|
||||
wait_for_healthy "$REDEPLOY_WAIT_HEALTHY_TIMEOUT" "$svc"
|
||||
else
|
||||
log INFO "${SUBINDENT}ℹ️ Keine Healthchecks → überspringe warten"
|
||||
fi
|
||||
|
||||
else
|
||||
log INFO " 🔄 Stack wird neu deployt (Trigger: ${changed_services[*]})"
|
||||
log INFO ""
|
||||
log INFO "${INDENT}♻️ Stack wird neu deployt (Trigger: ${changed_services[*]})"
|
||||
log INFO "${SUBINDENT}⏳ Deploy läuft..."
|
||||
|
||||
log INFO " ⏳ Deploy läuft..."
|
||||
if ! run_cmd docker compose up -d --remove-orphans --no-color >/dev/null 2>&1; then
|
||||
log ERROR " ❌ Stack Update fehlgeschlagen"
|
||||
if ! run_cmd docker compose up -d --remove-orphans >/dev/null 2>&1; then
|
||||
log ERROR "${SUBINDENT}❌ Stack Update fehlgeschlagen"
|
||||
error_flag=true
|
||||
else
|
||||
log INFO " ✔️ Stack erfolgreich aktualisiert"
|
||||
log INFO "${SUBINDENT}✅ Stack erfolgreich aktualisiert"
|
||||
|
||||
if [ "${REDEPLOY_WAIT:-0}" -gt 0 ]; then
|
||||
log INFO "${SUBINDENT}💤 Warte ${REDEPLOY_WAIT}s nach Deploy"
|
||||
sleep "$REDEPLOY_WAIT"
|
||||
fi
|
||||
|
||||
if [ "$REDEPLOY_WAIT_HEALTHY" = true ]; then
|
||||
wait_for_healthy "$REDEPLOY_WAIT_HEALTHY_TIMEOUT" "${changed_services[@]}"
|
||||
else
|
||||
log INFO "${SUBINDENT}ℹ️ Keine Healthchecks → überspringe warten"
|
||||
fi
|
||||
|
||||
# vorher gestoppte wieder stoppen
|
||||
for svc in "${services[@]}"; do
|
||||
if [ "${was_running[$svc]}" = 0 ]; then
|
||||
log INFO " ⏹️ Stoppe $svc (war vorher gestoppt)"
|
||||
log INFO "${SUBINDENT}⏹️ Stoppe $svc (war vorher gestoppt)"
|
||||
run_cmd docker compose stop "$svc" >/dev/null 2>&1 || true
|
||||
fi
|
||||
done
|
||||
|
||||
notify_stacks_updated+=("$stack (${changed_services[*]})")
|
||||
fi
|
||||
fi
|
||||
|
||||
|
||||
# =============================
|
||||
# Baum für ntfy bauen
|
||||
# =============================
|
||||
|
||||
if [ ${#update_lines[@]} -gt 0 ]; then
|
||||
stack_block=""
|
||||
|
||||
for i in "${!update_lines[@]}"; do
|
||||
IFS="|" read -r svc before after <<< "${update_lines[$i]}"
|
||||
|
||||
if [ "$i" -eq $((${#update_lines[@]} - 1)) ]; then
|
||||
prefix="└─"
|
||||
else
|
||||
log DEBUG " ✔️ Keine Updates"
|
||||
prefix="├─"
|
||||
fi
|
||||
|
||||
cd "$COMPOSE_DIR"
|
||||
if [ "$NTFY_SHOW_VERSIONS" = true ]; then
|
||||
stack_block+=$'\n'"$prefix **$svc** \`$before → $after\`"
|
||||
else
|
||||
stack_block+=$'\n'"$prefix **$svc**"
|
||||
fi
|
||||
done
|
||||
|
||||
done < <(find . -name "$COMPOSE_PATTERN" -print0 | sort -z)
|
||||
stack_tree["$stack"]="$stack_block"
|
||||
notify_stacks_updated+=("$stack")
|
||||
fi
|
||||
fi
|
||||
|
||||
cd "$PATH_COMPOSE_DIR"
|
||||
|
||||
stack_end=$(date +%s)
|
||||
log INFO "${INDENT}🕒 Dauer: $((stack_end - stack_start))s"
|
||||
|
||||
done < <(find . -name "$PATH_COMPOSE_PATTERN" -print0 | sort -z)
|
||||
|
||||
# =============================
|
||||
# Cleanup (mit Statistik)
|
||||
# Cleanup
|
||||
# =============================
|
||||
|
||||
freed_space="0"
|
||||
|
||||
if [ "$ENABLE_CLEANUP" = true ]; then
|
||||
log INFO ""
|
||||
|
||||
if [ "$CLEANUP_ENABLED" = true ]; then
|
||||
|
||||
if [ "$CLEANUP_ONLY_ON_UPDATE" = true ] && \
|
||||
[ ${#notify_stacks_updated[@]} -eq 0 ]; then
|
||||
@@ -334,42 +517,32 @@ if [ "$ENABLE_CLEANUP" = true ]; then
|
||||
else
|
||||
|
||||
before_size=$(get_docker_disk_usage)
|
||||
log INFO ""
|
||||
log INFO "${INDENT}🧹 Docker Cleanup läuft..."
|
||||
|
||||
log INFO "🧹 Docker Cleanup läuft..."
|
||||
|
||||
if [ "$CLEANUP_IMAGES" = true ]; then
|
||||
if [ "$CLEANUP_IMAGES_ENABLED" = true ]; then
|
||||
case "$CLEANUP_IMAGES_MODE" in
|
||||
unused)
|
||||
log INFO " → Entferne ungenutzte Images"
|
||||
run_cmd docker image prune -a -f >/dev/null 2>&1
|
||||
;;
|
||||
dangling)
|
||||
log INFO " → Entferne dangling Images"
|
||||
run_cmd docker image prune -f >/dev/null 2>&1
|
||||
;;
|
||||
unused) run_cmd docker image prune -a -f >/dev/null 2>&1 ;;
|
||||
dangling) run_cmd docker image prune -f >/dev/null 2>&1 ;;
|
||||
esac
|
||||
fi
|
||||
|
||||
if [ "$CLEANUP_CONTAINERS" = true ]; then
|
||||
log INFO " → Entferne gestoppte Container"
|
||||
if [ "$CLEANUP_CONTAINERS_ENABLED" = true ]; then
|
||||
run_cmd docker container prune -f >/dev/null 2>&1
|
||||
fi
|
||||
|
||||
if [ "$CLEANUP_VOLUMES" = true ]; then
|
||||
log WARN " → Entferne ungenutzte Volumes"
|
||||
if [ "$CLEANUP_VOLUMES_ENABLED" = true ]; then
|
||||
run_cmd docker volume prune -f >/dev/null 2>&1
|
||||
fi
|
||||
|
||||
if [ "$CLEANUP_NETWORKS" = true ]; then
|
||||
log INFO " → Entferne ungenutzte Netzwerke"
|
||||
if [ "$CLEANUP_NETWORKS_ENABLED" = true ]; then
|
||||
run_cmd docker network prune -f >/dev/null 2>&1
|
||||
fi
|
||||
|
||||
after_size=$(get_docker_disk_usage)
|
||||
freed_space=$((after_size < before_size ? before_size - after_size : 0))
|
||||
|
||||
freed_space=$(awk "BEGIN {print $before_size - $after_size}")
|
||||
|
||||
log INFO "✔️ Cleanup abgeschlossen (freigegeben: ${freed_space} MB)"
|
||||
log INFO "${INDENT}✅ Cleanup abgeschlossen (${freed_space} MB freigegeben)"
|
||||
fi
|
||||
fi
|
||||
|
||||
@@ -379,50 +552,46 @@ fi
|
||||
|
||||
if [ "$NTFY_ENABLED" = true ]; then
|
||||
|
||||
msg="Docker Compose Update Report"
|
||||
msg=""
|
||||
|
||||
if [ ${#notify_stacks_updated[@]} -gt 0 ]; then
|
||||
msg+=$'\n\n🔄 Aktualisierte Stacks'
|
||||
for s in "${notify_stacks_updated[@]}"; do
|
||||
msg+=$'\n - '"$s"
|
||||
if [ ${#stack_tree[@]} -gt 0 ]; then
|
||||
msg+=$'#### 🔄 Stack Updates\n'
|
||||
|
||||
for stack in $(printf "%s\n" "${!stack_tree[@]}" | sort); do
|
||||
msg+=$'\n'"$stack"
|
||||
msg+="${stack_tree[$stack]}"
|
||||
# msg+=$'\n' # Leerzeile nach jedem Stack
|
||||
done
|
||||
fi
|
||||
|
||||
if [ ${#notify_excluded_updates[@]} -gt 0 ]; then
|
||||
msg+=$'\n\n⏭️ Excluded (Update verfügbar)'
|
||||
msg+=$'\n\n#### ⏭️ Excluded (Update verfügbar)'
|
||||
for s in "${notify_excluded_updates[@]}"; do
|
||||
msg+=$'\n - '"$s"
|
||||
done
|
||||
fi
|
||||
|
||||
if [ ${#stack_tree[@]} -eq 0 ] && \
|
||||
[ ${#notify_excluded_updates[@]} -eq 0 ]; then
|
||||
msg+=$'\n\n#### ✔️ Alles aktuell'
|
||||
fi
|
||||
|
||||
if [ "$freed_space" != "0" ]; then
|
||||
msg+=$'\n\n#### 🧹 Cleanup: '"${freed_space} MB freigegeben"
|
||||
fi
|
||||
|
||||
if [ "$error_flag" = true ]; then
|
||||
msg+=$'\n\n❗ Fehler sind aufgetreten – Logs prüfen'
|
||||
PRIORITY=5
|
||||
elif [ ${#notify_stacks_updated[@]} -gt 0 ]; then
|
||||
elif [ ${#stack_tree[@]} -gt 0 ]; then
|
||||
PRIORITY=3
|
||||
else
|
||||
PRIORITY=1
|
||||
fi
|
||||
|
||||
if [ ${#notify_stacks_updated[@]} -eq 0 ] && \
|
||||
[ ${#notify_excluded_updates[@]} -eq 0 ]; then
|
||||
msg+=$'\n\n✔️ Alles aktuell'
|
||||
fi
|
||||
|
||||
if [ "$freed_space" != "0" ]; then
|
||||
msg+=$'\n\n🧹 Cleanup: '"${freed_space} MB freigegeben"
|
||||
fi
|
||||
|
||||
if [ "$NTFY_ONLY_ON_CHANGES" = false ] || \
|
||||
[ ${#notify_stacks_updated[@]} -gt 0 ] || \
|
||||
[ ${#notify_excluded_updates[@]} -gt 0 ] || \
|
||||
[ "$error_flag" = true ]; then
|
||||
|
||||
send_ntfy "$msg" "$PRIORITY"
|
||||
log INFO "ntfy Nachricht gesendet (prio=$PRIORITY)"
|
||||
else
|
||||
log INFO "keine Änderungen, keine ntfy Nachricht"
|
||||
fi
|
||||
log INFO "📨 ntfy Nachricht gesendet (prio=$PRIORITY)"
|
||||
fi
|
||||
|
||||
script_end=$(date +%s)
|
||||
log INFO "🕒 Gesamtzeit: $((script_end - script_start))s"
|
||||
log INFO "==== Update beendet ===="
|
||||
Reference in New Issue
Block a user