4 Commits

Author SHA1 Message Date
martin c05f323605 feat: Trockenlauf als Standard-Aufruf, --execute/-e für echten Sync
Ohne Flags führt darktable_sync.sh jetzt einen Trockenlauf durch:
- Banner mit Hinweis und Bestätigungsabfrage vor dem Start
- rsync läuft mit --dry-run (keine Dateiänderungen)
- Keine destruktiven Operationen: kein Backup, kein Token-Schreiben,
  kein sync_pending entfernen
- Zusammenfassung nach Richtung (Upload/Download) und Aktion
  (neu/aktualisiert/gelöscht) aufgeschlüsselt
- Optionale Detailansicht: Dateien gruppiert nach Typ (Foto, XMP,
  Datenbank, Video, Sonstiges)

Mit --execute oder -e wird der echte Sync wie bisher ausgeführt.
Desktop-Entry und Systemd-Service auf --execute aktualisiert.

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-20 16:24:31 +02:00
martin 0c5774f695 install.sh: interaktive Abfragen bei Installation verbessert
- .env-Verschiebung aus Projektverzeichnis wird angeboten und bei Bestätigung automatisch ausgeführt
- Lokales Foto-Verzeichnis wird interaktiv abgefragt (mit Hinweis falls es bereits existiert)
- Ausführlicher Hinweis nach Anlegen der Default-.env mit Pflichtfeldern und nächsten Schritten
- bats-Hinweis entfernt (nur für Entwickler relevant)
- Tests: Umlaut-Mismatch in security.bats behoben, teardown() für Lock-Isolation ergänzt

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-20 12:47:10 +02:00
martin bc63739399 Merge pull request 'Robustheit: Error-Handling, Validation und Strukturverbesserungen' (#3) from feat/improve-sync-robustness into main 2026-04-19 21:05:41 +02:00
martin c4a9b4a33d Robustheit: Error-Handling, Validation und Strukturverbesserungen
- Neue darktable_common.sh mit gemeinsamen Funktionen (Logging, Validierung, Lock-Management)
- Verbesserte Fehlerbehandlung und aussagekräftige Error-Messages
- Explizite Validierung von SSH-Schlüssel, Pfaden und Konfiguration beim Start
- Sperrmechanismus zur Verhinderung paralleler Sync-Instanzen
- Bessere Strukturierung des Sync-Prozesses mit sauberer Fehlertoleranz

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-04-19 21:05:14 +02:00
9 changed files with 380 additions and 79 deletions
+1 -1
View File
@@ -2,6 +2,6 @@
Type=Application Type=Application
Name=Darktable Sync Name=Darktable Sync
Comment=Nur Synchronisation ausfuehren ohne Darktable zu starten Comment=Nur Synchronisation ausfuehren ohne Darktable zu starten
Exec=/home/%u/.local/bin/darktable_sync.sh --with-notify-start-stop Exec=/home/%u/.local/bin/darktable_sync.sh --execute --with-notify-start-stop
Terminal=false Terminal=false
Categories=Graphics;Photography; Categories=Graphics;Photography;
+42 -7
View File
@@ -37,9 +37,18 @@ CONFIG_ENV="$CONFIG_DIR/.env"
if [[ -f "$ENV_FILE" ]]; then if [[ -f "$ENV_FILE" ]]; then
echo "Hinweis: .env im Projektverzeichnis gefunden." echo "Hinweis: .env im Projektverzeichnis gefunden."
echo " Bitte nach $CONFIG_ENV verschieben:" read -r -p " Jetzt nach $CONFIG_ENV verschieben? [J/n]: " MOVE_ENV
if [[ "${MOVE_ENV,,}" != "n" ]]; then
mkdir -p "$CONFIG_DIR"
cp "$ENV_FILE" "$CONFIG_ENV"
chmod 600 "$CONFIG_ENV"
rm "$ENV_FILE"
echo " Erledigt: .env wurde verschoben."
else
echo " Nicht verschoben. Bitte manuell ausfuehren:"
echo " cp .env $CONFIG_ENV && chmod 600 $CONFIG_ENV" echo " cp .env $CONFIG_ENV && chmod 600 $CONFIG_ENV"
fi fi
fi
if [[ -f "$CONFIG_ENV" ]]; then if [[ -f "$CONFIG_ENV" ]]; then
echo "Konfiguration laden aus $CONFIG_ENV..." echo "Konfiguration laden aus $CONFIG_ENV..."
@@ -49,6 +58,18 @@ if [[ -f "$CONFIG_ENV" ]]; then
set +a set +a
fi fi
### Lokales Foto-Verzeichnis interaktiv abfragen
echo ""
if [[ -d "$LOCAL_PHOTO_DIR" ]]; then
read -r -p "Lokales Foto-Verzeichnis [${LOCAL_PHOTO_DIR}] (Verzeichnis existiert bereits): " INPUT_PHOTO_DIR
else
read -r -p "Lokales Foto-Verzeichnis [${LOCAL_PHOTO_DIR}]: " INPUT_PHOTO_DIR
fi
if [[ -n "$INPUT_PHOTO_DIR" ]]; then
LOCAL_PHOTO_DIR="$INPUT_PHOTO_DIR"
fi
### Konfiguration anzeigen ### Konfiguration anzeigen
echo "" echo ""
@@ -84,11 +105,6 @@ if ! command -v zenity >/dev/null 2>&1 && ! command -v kdialog >/dev/null 2>&1;
echo " (Ohne Dialog-Tool wird ein Text-Fallback verwendet)" echo " (Ohne Dialog-Tool wird ein Text-Fallback verwendet)"
fi fi
if ! command -v bats >/dev/null 2>&1; then
echo "Hinweis: 'bats' nicht gefunden (nur fuer Tests benoetigt)."
echo " sudo apt install bats"
fi
### Verzeichnisse pruefen ### Verzeichnisse pruefen
if [ ! -d "$LOCAL_PHOTO_DIR" ]; then if [ ! -d "$LOCAL_PHOTO_DIR" ]; then
@@ -166,8 +182,27 @@ if [ ! -f "$CONFIG_ENV" ]; then
cp "$SCRIPT_DIR/.env.example" "$CONFIG_ENV" cp "$SCRIPT_DIR/.env.example" "$CONFIG_ENV"
chmod 600 "$CONFIG_ENV" chmod 600 "$CONFIG_ENV"
echo "" echo ""
echo "WICHTIG: Konfiguration anpassen:" echo "=========================================================="
echo "WICHTIG: Konfiguration anpassen, bevor du Darktable startest"
echo "=========================================================="
echo ""
echo "Eine Vorlage wurde angelegt:"
echo " $CONFIG_ENV"
echo ""
echo "Mindestens diese Felder musst du eintragen:"
echo " SERVER_USER - dein SSH-Benutzer auf dem Server"
echo " SERVER_IP - IP-Adresse oder Hostname des Servers"
echo " SERVER_DB_DIR - Pfad zur Darktable-Datenbank auf dem Server"
echo " SERVER_PHOTO_DIR - Pfad zum Fotoverzeichnis auf dem Server"
echo ""
echo "LOCAL_PHOTO_DIR ist bereits auf '${LOCAL_PHOTO_DIR}' gesetzt."
echo ""
echo "Jetzt bearbeiten:"
echo " nano $CONFIG_ENV" echo " nano $CONFIG_ENV"
echo ""
echo "Danach install.sh erneut ausfuehren, damit die Verbindung"
echo "zum Server geprueft wird."
echo "=========================================================="
fi fi
### Desktop-Shortcuts installieren ### Desktop-Shortcuts installieren
+112
View File
@@ -88,11 +88,69 @@ log_error() {
echo "FEHLER: $*" >&2 echo "FEHLER: $*" >&2
} }
classify_filetype() {
local file="$1"
local ext="${file##*.}"; ext="${ext,,}"
case "$ext" in
jpg|jpeg|png|tif|tiff|dng|cr2|cr3|nef|arw|orf|rw2|raf|raw) echo "Foto" ;;
xmp) echo "XMP" ;;
db|bak) echo "Datenbank" ;;
mp4|mov|avi|mkv|mts|m2ts) echo "Video" ;;
*) echo "Sonstiges" ;;
esac
}
format_rsync_details() {
local log_file="$1" direction_label="$2" direction="$3"
[ -f "$log_file" ] || return 0
local prefix; [ "$direction" = "up" ] && prefix=">f" || prefix="<f"
local label pattern files
while IFS=: read -r label pattern; do
files=$(grep -E "$pattern" "$log_file" 2>/dev/null \
| sed 's/^[^ ]* *//' | sort) || true
[ -n "$files" ] || continue
local typ typed
for typ in Foto XMP Datenbank Video Sonstiges; do
typed=$(echo "$files" | while IFS= read -r f; do
[ "$(classify_filetype "$f")" = "$typ" ] && echo " $f"
done)
[ -n "$typed" ] || continue
log_step "$direction_label $typ ($label)"
echo "$typed"
done
done <<EOF
neu:^${prefix}[+]{9}
aktualisiert:^${prefix}[^+]
gelöscht:^\*deleting
EOF
}
confirm_dry_run() {
[ "${DRY_RUN_SKIP_CONFIRM:-0}" = "1" ] && return 0
ask_user "Darktable Sync Trockenlauf" \
"Trockenlauf starten?\n\nEs werden keine Dateien verändert oder übertragen."
}
ssh_server() { ssh_server() {
ssh -o ConnectTimeout=5 -o BatchMode=yes \ ssh -o ConnectTimeout=5 -o BatchMode=yes \
-p "$SERVER_SSH_PORT" "$SERVER_USER@$SERVER_IP" "$@" -p "$SERVER_SSH_PORT" "$SERVER_USER@$SERVER_IP" "$@"
} }
# Liefert den Unix-Timestamp (mtime) von library.db auf dem Server, oder "0" wenn nicht vorhanden.
server_db_mtime() {
ssh_server "stat -c '%Y' '$SERVER_DB_DIR/library.db' 2>/dev/null || echo 0"
}
save_sync_token() {
echo "$1" > "$CONFIG_DIR/sync_token"
}
read_sync_token() {
cat "$CONFIG_DIR/sync_token" 2>/dev/null || echo ""
}
server_reachable() { server_reachable() {
ssh_server true 2>/dev/null ssh_server true 2>/dev/null
} }
@@ -111,3 +169,57 @@ ask_user() {
return $? return $?
fi fi
} }
# Fragt den User wie mit einem Sync-Token-Konflikt umgegangen werden soll.
# Gibt "download", "upload" oder "abort" aus.
ask_conflict_resolution() {
local TITLE="Darktable Sync Konflikt"
local EXPLAIN="Ein anderer Rechner hat die Datenbank seit deinem letzten Sync verändert.\nDeine lokalen Änderungen wurden noch NICHT auf den Server übertragen.\n\nWas soll passieren?"
if command -v zenity &>/dev/null; then
local choice
choice=$(zenity --list \
--title="$TITLE" \
--text="$EXPLAIN" \
--radiolist \
--column="" --column="Aktion" --column="Beschreibung" \
TRUE "Herunterladen" "Server-Stand übernehmen (empfohlen)" \
FALSE "Hochladen erzwingen" "Lokale Version auf Server schreiben Server-Änderungen gehen verloren!" \
FALSE "Abbrechen" "Nichts tun Sync wird übersprungen" \
--width=520 --height=260 2>/dev/null) || true
case "$choice" in
"Hochladen erzwingen") echo "upload" ;;
"Abbrechen") echo "abort" ;;
*) echo "download" ;;
esac
elif command -v kdialog &>/dev/null; then
local btn
btn=$(kdialog --title "$TITLE" \
--menu "$EXPLAIN" \
download "Herunterladen (empfohlen)" \
upload "Hochladen erzwingen (Server-Änderungen gehen verloren!)" \
abort "Abbrechen" 2>/dev/null) || true
case "$btn" in
upload|abort) echo "$btn" ;;
*) echo "download" ;;
esac
else
echo ""
echo "=== $TITLE ==="
echo "Ein anderer Rechner hat die Datenbank seit deinem letzten Sync verändert."
echo "Deine lokalen Änderungen wurden noch NICHT auf den Server übertragen."
echo ""
echo " 1) Herunterladen (empfohlen) Server-Stand übernehmen"
echo " 2) Hochladen erzwingen lokale Version gewinnt, Server-Änderungen gehen verloren"
echo " 3) Abbrechen"
local ans
read -r -p "Auswahl [1/2/3, Standard: 1]: " ans || true
case "$ans" in
2) echo "upload" ;;
3) echo "abort" ;;
*) echo "download" ;;
esac
fi
}
+106 -16
View File
@@ -23,6 +23,15 @@ log " Fotos Server:$SERVER_PHOTO_DIR"
export DISPLAY="${DISPLAY:-:0}" export DISPLAY="${DISPLAY:-:0}"
DRY_RUN=true
SHOW_NOTIFY_START_STOP=false
for arg in "$@"; do
case "$arg" in
--execute|-e) DRY_RUN=false ;;
--with-notify-start-stop) SHOW_NOTIFY_START_STOP=true ;;
esac
done
count_synced_files() { count_synced_files() {
local log_file="$1" direction="$2" count=0 local log_file="$1" direction="$2" count=0
case "$direction" in case "$direction" in
@@ -54,9 +63,15 @@ echo "$$" > "$LOCKPID"
log "Lock erworben (PID $$)." log "Lock erworben (PID $$)."
trap 'rm -f "${TMPFILES[@]}" "$LOCKPID"; rmdir "$LOCKDIR" 2>/dev/null || true; log "Lock freigegeben."' EXIT trap 'rm -f "${TMPFILES[@]}" "$LOCKPID"; rmdir "$LOCKDIR" 2>/dev/null || true; log "Lock freigegeben."' EXIT
SHOW_NOTIFY_START_STOP=false if [ "$DRY_RUN" = true ]; then
if [[ "${1:-}" == "--with-notify-start-stop" ]]; then log_step "TROCKENLAUF keine Änderungen werden vorgenommen"
SHOW_NOTIFY_START_STOP=true log "Dieser Aufruf zeigt nur, was synchronisiert werden würde."
log "Für echten Sync: $(basename "$0") --execute oder -e"
log ""
if ! confirm_dry_run; then
log "Trockenlauf abgebrochen."
exit 0
fi
fi fi
log "Prüfen ob Darktable läuft..." log "Prüfen ob Darktable läuft..."
@@ -117,22 +132,67 @@ if [ -n "$SERVER_VERSION" ]; then
log "Versionen übereinstimmend ($LOCAL_MM)." log "Versionen übereinstimmend ($LOCAL_MM)."
fi fi
log "Sync-Token prüfen..."
SAVED_TOKEN=$(read_sync_token)
SERVER_TOKEN=$(server_db_mtime)
log " Gespeicherter Token: ${SAVED_TOKEN:-keiner (erster Sync)}"
log " Aktueller Server-Token: $SERVER_TOKEN"
UPLOAD_ALLOWED=true
if [ -n "$SAVED_TOKEN" ] && [ "$SAVED_TOKEN" != "$SERVER_TOKEN" ]; then
log "WARNUNG: Token-Konflikt (gespeichert=$SAVED_TOKEN, server=$SERVER_TOKEN) Benutzer wird gefragt."
RESOLUTION=$(ask_conflict_resolution)
log "Benutzerentscheidung: $RESOLUTION"
case "$RESOLUTION" in
upload)
log "Upload erzwungen lokale Version überschreibt Server."
;;
abort)
log "Sync abgebrochen durch Benutzer."
exit 0
;;
*)
log "Nur Download Server-Stand wird übernommen."
UPLOAD_ALLOWED=false
;;
esac
else
log "Token stimmt überein Upload erlaubt."
fi
if [ "$DRY_RUN" = false ]; then
log_step "Datenbank-Backup" log_step "Datenbank-Backup"
log " $LOCAL_DARKTABLE_DB_DIR/library.db → library.db.bak" log " $LOCAL_DARKTABLE_DB_DIR/library.db → library.db.bak"
cp "$LOCAL_DARKTABLE_DB_DIR/library.db" "$LOCAL_DARKTABLE_DB_DIR/library.db.bak" cp "$LOCAL_DARKTABLE_DB_DIR/library.db" "$LOCAL_DARKTABLE_DB_DIR/library.db.bak"
log " $LOCAL_DARKTABLE_DB_DIR/data.db → data.db.bak" log " $LOCAL_DARKTABLE_DB_DIR/data.db → data.db.bak"
cp "$LOCAL_DARKTABLE_DB_DIR/data.db" "$LOCAL_DARKTABLE_DB_DIR/data.db.bak" cp "$LOCAL_DARKTABLE_DB_DIR/data.db" "$LOCAL_DARKTABLE_DB_DIR/data.db.bak"
log "Backup abgeschlossen." log "Backup abgeschlossen."
fi
SYNC_LOG=$(mktemp) SYNC_LOG=$(mktemp)
TMPFILES+=("$SYNC_LOG") TMPFILES+=("$SYNC_LOG")
log_step "Upload: Datenbank"
log " Quelle: $LOCAL_DARKTABLE_DB_DIR/"
log " Ziel: $SERVER_USER@$SERVER_IP:$SERVER_DB_DIR/"
UPLOAD_LOG_DB=$(mktemp) UPLOAD_LOG_DB=$(mktemp)
TMPFILES+=("$UPLOAD_LOG_DB") TMPFILES+=("$UPLOAD_LOG_DB")
UPLOAD_LOG_PHOTOS=$(mktemp)
TMPFILES+=("$UPLOAD_LOG_PHOTOS")
DOWNLOAD_LOG_DB=$(mktemp)
TMPFILES+=("$DOWNLOAD_LOG_DB")
DOWNLOAD_LOG_PHOTOS=$(mktemp)
TMPFILES+=("$DOWNLOAD_LOG_PHOTOS")
RSYNC_DRY_FLAG=()
[ "$DRY_RUN" = true ] && RSYNC_DRY_FLAG=(--dry-run)
[ "$DRY_RUN" = true ] && DRY_SUFFIX=" (Trockenlauf)" || DRY_SUFFIX=""
SENT_DB=0
SENT_PHOTOS=0
if [ "$UPLOAD_ALLOWED" = true ]; then
log_step "Upload: Datenbank${DRY_SUFFIX}"
log " Quelle: $LOCAL_DARKTABLE_DB_DIR/"
log " Ziel: $SERVER_USER@$SERVER_IP:$SERVER_DB_DIR/"
if ! rsync -uavh --itemize-changes \ if ! rsync -uavh --itemize-changes \
"${RSYNC_DRY_FLAG[@]}" \
--exclude '*.lock' \ --exclude '*.lock' \
--exclude 'darktable_version' \ --exclude 'darktable_version' \
-e "ssh -p $SERVER_SSH_PORT" \ -e "ssh -p $SERVER_SSH_PORT" \
@@ -145,12 +205,11 @@ fi
SENT_DB=$(count_synced_files "$UPLOAD_LOG_DB" "up") SENT_DB=$(count_synced_files "$UPLOAD_LOG_DB" "up")
log "Datenbank-Upload abgeschlossen: $SENT_DB Datei(en) übertragen." log "Datenbank-Upload abgeschlossen: $SENT_DB Datei(en) übertragen."
log_step "Upload: Fotos" log_step "Upload: Fotos${DRY_SUFFIX}"
log " Quelle: $LOCAL_PHOTO_DIR/" log " Quelle: $LOCAL_PHOTO_DIR/"
log " Ziel: $SERVER_USER@$SERVER_IP:$SERVER_PHOTO_DIR/" log " Ziel: $SERVER_USER@$SERVER_IP:$SERVER_PHOTO_DIR/"
UPLOAD_LOG_PHOTOS=$(mktemp)
TMPFILES+=("$UPLOAD_LOG_PHOTOS")
if ! rsync -uavh --itemize-changes \ if ! rsync -uavh --itemize-changes \
"${RSYNC_DRY_FLAG[@]}" \
--exclude '*.lock' \ --exclude '*.lock' \
-e "ssh -p $SERVER_SSH_PORT" \ -e "ssh -p $SERVER_SSH_PORT" \
"$LOCAL_PHOTO_DIR/" "$SERVER_USER@$SERVER_IP:$SERVER_PHOTO_DIR/" \ "$LOCAL_PHOTO_DIR/" "$SERVER_USER@$SERVER_IP:$SERVER_PHOTO_DIR/" \
@@ -161,13 +220,15 @@ if ! rsync -uavh --itemize-changes \
fi fi
SENT_PHOTOS=$(count_synced_files "$UPLOAD_LOG_PHOTOS" "up") SENT_PHOTOS=$(count_synced_files "$UPLOAD_LOG_PHOTOS" "up")
log "Foto-Upload abgeschlossen: $SENT_PHOTOS Datei(en) übertragen." log "Foto-Upload abgeschlossen: $SENT_PHOTOS Datei(en) übertragen."
else
log "Upload übersprungen (Token-Konflikt)."
fi
log_step "Download: Datenbank" log_step "Download: Datenbank${DRY_SUFFIX}"
log " Quelle: $SERVER_USER@$SERVER_IP:$SERVER_DB_DIR/" log " Quelle: $SERVER_USER@$SERVER_IP:$SERVER_DB_DIR/"
log " Ziel: $LOCAL_DARKTABLE_DB_DIR/" log " Ziel: $LOCAL_DARKTABLE_DB_DIR/"
DOWNLOAD_LOG_DB=$(mktemp)
TMPFILES+=("$DOWNLOAD_LOG_DB")
if ! rsync -uavh --itemize-changes \ if ! rsync -uavh --itemize-changes \
"${RSYNC_DRY_FLAG[@]}" \
--exclude '*.lock' \ --exclude '*.lock' \
-e "ssh -p $SERVER_SSH_PORT" \ -e "ssh -p $SERVER_SSH_PORT" \
"$SERVER_USER@$SERVER_IP:$SERVER_DB_DIR/" "$LOCAL_DARKTABLE_DB_DIR/" \ "$SERVER_USER@$SERVER_IP:$SERVER_DB_DIR/" "$LOCAL_DARKTABLE_DB_DIR/" \
@@ -179,12 +240,11 @@ fi
RECEIVED_DB=$(count_synced_files "$DOWNLOAD_LOG_DB" "down") RECEIVED_DB=$(count_synced_files "$DOWNLOAD_LOG_DB" "down")
log "Datenbank-Download abgeschlossen: $RECEIVED_DB Datei(en) empfangen." log "Datenbank-Download abgeschlossen: $RECEIVED_DB Datei(en) empfangen."
log_step "Download: Fotos" log_step "Download: Fotos${DRY_SUFFIX}"
log " Quelle: $SERVER_USER@$SERVER_IP:$SERVER_PHOTO_DIR/" log " Quelle: $SERVER_USER@$SERVER_IP:$SERVER_PHOTO_DIR/"
log " Ziel: $LOCAL_PHOTO_DIR/" log " Ziel: $LOCAL_PHOTO_DIR/"
DOWNLOAD_LOG_PHOTOS=$(mktemp)
TMPFILES+=("$DOWNLOAD_LOG_PHOTOS")
if ! rsync -uavh --itemize-changes \ if ! rsync -uavh --itemize-changes \
"${RSYNC_DRY_FLAG[@]}" \
--exclude '*.lock' \ --exclude '*.lock' \
-e "ssh -p $SERVER_SSH_PORT" \ -e "ssh -p $SERVER_SSH_PORT" \
"$SERVER_USER@$SERVER_IP:$SERVER_PHOTO_DIR/" "$LOCAL_PHOTO_DIR/" \ "$SERVER_USER@$SERVER_IP:$SERVER_PHOTO_DIR/" "$LOCAL_PHOTO_DIR/" \
@@ -196,15 +256,44 @@ fi
RECEIVED_PHOTOS=$(count_synced_files "$DOWNLOAD_LOG_PHOTOS" "down") RECEIVED_PHOTOS=$(count_synced_files "$DOWNLOAD_LOG_PHOTOS" "down")
log "Foto-Download abgeschlossen: $RECEIVED_PHOTOS Datei(en) empfangen." log "Foto-Download abgeschlossen: $RECEIVED_PHOTOS Datei(en) empfangen."
if [ "$DRY_RUN" = false ]; then
NEW_TOKEN=$(server_db_mtime)
save_sync_token "$NEW_TOKEN"
log "Sync-Token gespeichert: $NEW_TOKEN"
log "Versionsdatei aktualisieren: $LOCAL_DARKTABLE_DB_DIR/darktable_version" log "Versionsdatei aktualisieren: $LOCAL_DARKTABLE_DB_DIR/darktable_version"
echo "$LOCAL_VERSION" > "$LOCAL_DARKTABLE_DB_DIR/darktable_version" echo "$LOCAL_VERSION" > "$LOCAL_DARKTABLE_DB_DIR/darktable_version"
rm -f "$CONFIG_DIR/sync_pending" rm -f "$CONFIG_DIR/sync_pending"
log "sync_pending entfernt." log "sync_pending entfernt."
fi
TOTAL_SENT=$((SENT_DB + SENT_PHOTOS)) TOTAL_SENT=$((SENT_DB + SENT_PHOTOS))
TOTAL_RECEIVED=$((RECEIVED_DB + RECEIVED_PHOTOS)) TOTAL_RECEIVED=$((RECEIVED_DB + RECEIVED_PHOTOS))
if [ "$DRY_RUN" = true ]; then
UP_NEW=$( cat "$UPLOAD_LOG_DB" "$UPLOAD_LOG_PHOTOS" 2>/dev/null | grep -cE '^>f[+]{9}' || echo 0)
UP_UPD=$( cat "$UPLOAD_LOG_DB" "$UPLOAD_LOG_PHOTOS" 2>/dev/null | grep -E '^>f' | grep -cvE '^>f[+]{9}' || echo 0)
UP_DEL=$( cat "$UPLOAD_LOG_DB" "$UPLOAD_LOG_PHOTOS" 2>/dev/null | grep -cE '^\*deleting' || echo 0)
DN_NEW=$( cat "$DOWNLOAD_LOG_DB" "$DOWNLOAD_LOG_PHOTOS" 2>/dev/null | grep -cE '^<f[+]{9}' || echo 0)
DN_UPD=$( cat "$DOWNLOAD_LOG_DB" "$DOWNLOAD_LOG_PHOTOS" 2>/dev/null | grep -E '^<f' | grep -cvE '^<f[+]{9}' || echo 0)
DN_DEL=$( cat "$DOWNLOAD_LOG_DB" "$DOWNLOAD_LOG_PHOTOS" 2>/dev/null | grep -cE '^\*deleting' || echo 0)
log_step "Trockenlauf-Ergebnis"
log " Upload: $UP_NEW neu | $UP_UPD aktualisiert | $UP_DEL gelöscht"
log " Download: $DN_NEW neu | $DN_UPD aktualisiert | $DN_DEL gelöscht"
if [ "$((TOTAL_SENT + TOTAL_RECEIVED))" -gt 0 ]; then
if ask_user "Details" "Details der zu übertragenden Dateien anzeigen?"; then
format_rsync_details "$UPLOAD_LOG_DB" "Upload" "up"
format_rsync_details "$UPLOAD_LOG_PHOTOS" "Upload" "up"
format_rsync_details "$DOWNLOAD_LOG_DB" "Download" "down"
format_rsync_details "$DOWNLOAD_LOG_PHOTOS" "Download" "down"
fi
else
log "Keine Änderungen alles aktuell."
fi
else
log_step "Sync abgeschlossen" log_step "Sync abgeschlossen"
log " Hochgeladen: $TOTAL_SENT ($SENT_DB DB + $SENT_PHOTOS Fotos)" log " Hochgeladen: $TOTAL_SENT ($SENT_DB DB + $SENT_PHOTOS Fotos)"
log " Heruntergeladen: $TOTAL_RECEIVED ($RECEIVED_DB DB + $RECEIVED_PHOTOS Fotos)" log " Heruntergeladen: $TOTAL_RECEIVED ($RECEIVED_DB DB + $RECEIVED_PHOTOS Fotos)"
@@ -219,3 +308,4 @@ fi
if [ "$SHOW_NOTIFY_START_STOP" = true ]; then if [ "$SHOW_NOTIFY_START_STOP" = true ]; then
notify-send "Darktable Sync" "Sync abgeschlossen." -t 3000 notify-send "Darktable Sync" "Sync abgeschlossen." -t 3000
fi fi
fi
+1 -1
View File
@@ -3,4 +3,4 @@ Description=Darktable Sync (manueller Trigger)
[Service] [Service]
Type=oneshot Type=oneshot
ExecStart=%h/.local/bin/darktable_sync.sh ExecStart=%h/.local/bin/darktable_sync.sh --execute
+59 -7
View File
@@ -9,31 +9,34 @@ setup() {
mkdir -p "$HOME/.config/darktable" mkdir -p "$HOME/.config/darktable"
touch "$HOME/.config/darktable/library.db" touch "$HOME/.config/darktable/library.db"
touch "$HOME/.config/darktable/data.db" touch "$HOME/.config/darktable/data.db"
rm -f "$HOME/.config/darktable/"*.bak
mkdir -p "$HOME/Pictures" mkdir -p "$HOME/Pictures"
export DISPLAY=:99 export DISPLAY=:99
} }
# --- Bestehende Tests (echter Sync via --execute) ---
@test "sync_pending wird gesetzt wenn Server nicht erreichbar" { @test "sync_pending wird gesetzt wenn Server nicht erreichbar" {
run_with_stubs env SSH_STUB_FAIL=1 bash "$SYNC_SCRIPT" run_with_stubs env SSH_STUB_FAIL=1 bash "$SYNC_SCRIPT" --execute
[ "$status" -eq 0 ] [ "$status" -eq 0 ]
[ -f "$CONFIG_DIR/sync_pending" ] [ -f "$CONFIG_DIR/sync_pending" ]
} }
@test "sync_pending wird entfernt bei erfolgreichem Sync" { @test "sync_pending wird entfernt bei erfolgreichem Sync" {
touch "$CONFIG_DIR/sync_pending" touch "$CONFIG_DIR/sync_pending"
run_with_stubs env SSH_STUB_FAIL=0 bash "$SYNC_SCRIPT" run_with_stubs env SSH_STUB_FAIL=0 bash "$SYNC_SCRIPT" --execute
[ "$status" -eq 0 ] [ "$status" -eq 0 ]
[ ! -f "$CONFIG_DIR/sync_pending" ] [ ! -f "$CONFIG_DIR/sync_pending" ]
} }
@test "sync_pending wird gesetzt wenn rsync fehlschlaegt" { @test "sync_pending wird gesetzt wenn rsync fehlschlaegt" {
run_with_stubs env SSH_STUB_FAIL=0 RSYNC_STUB_FAIL=1 bash "$SYNC_SCRIPT" run_with_stubs env SSH_STUB_FAIL=0 RSYNC_STUB_FAIL=1 bash "$SYNC_SCRIPT" --execute
[ "$status" -eq 1 ] [ "$status" -eq 1 ]
[ -f "$CONFIG_DIR/sync_pending" ] [ -f "$CONFIG_DIR/sync_pending" ]
} }
@test "DB-Backup wird vor Download erstellt" { @test "DB-Backup wird vor Download erstellt" {
run_with_stubs env SSH_STUB_FAIL=0 bash "$SYNC_SCRIPT" run_with_stubs env SSH_STUB_FAIL=0 bash "$SYNC_SCRIPT" --execute
[ "$status" -eq 0 ] [ "$status" -eq 0 ]
[ -f "$HOME/.config/darktable/library.db.bak" ] [ -f "$HOME/.config/darktable/library.db.bak" ]
[ -f "$HOME/.config/darktable/data.db.bak" ] [ -f "$HOME/.config/darktable/data.db.bak" ]
@@ -41,19 +44,68 @@ setup() {
@test "Versionskonflikt: gleiche Major.Minor gibt kein Exit 1" { @test "Versionskonflikt: gleiche Major.Minor gibt kein Exit 1" {
run_with_stubs env SSH_STUB_FAIL=0 SSH_STUB_OUTPUT="this is darktable 5.0.1" \ run_with_stubs env SSH_STUB_FAIL=0 SSH_STUB_OUTPUT="this is darktable 5.0.1" \
DARKTABLE_STUB_VERSION="5.0.1" bash "$SYNC_SCRIPT" DARKTABLE_STUB_VERSION="5.0.1" DRY_RUN_SKIP_CONFIRM=1 bash "$SYNC_SCRIPT"
[ "$status" -eq 0 ] [ "$status" -eq 0 ]
} }
@test "Versionskonflikt: andere Major.Minor gibt Exit 1" { @test "Versionskonflikt: andere Major.Minor gibt Exit 1" {
run_with_stubs env SSH_STUB_FAIL=0 SSH_STUB_OUTPUT="this is darktable 4.8.0" \ run_with_stubs env SSH_STUB_FAIL=0 SSH_STUB_OUTPUT="this is darktable 4.8.0" \
DARKTABLE_STUB_VERSION="5.0.0" bash "$SYNC_SCRIPT" DARKTABLE_STUB_VERSION="5.0.0" DRY_RUN_SKIP_CONFIRM=1 bash "$SYNC_SCRIPT"
[ "$status" -eq 1 ] [ "$status" -eq 1 ]
[ -f "$CONFIG_DIR/sync_pending" ] [ -f "$CONFIG_DIR/sync_pending" ]
} }
@test "Lockdir wird nach Abschluss entfernt" { @test "Lockdir wird nach Abschluss entfernt" {
run_with_stubs env SSH_STUB_FAIL=0 bash "$SYNC_SCRIPT" run_with_stubs env SSH_STUB_FAIL=0 bash "$SYNC_SCRIPT" --execute
[ "$status" -eq 0 ] [ "$status" -eq 0 ]
[ ! -d "$CONFIG_DIR/sync.lock" ] [ ! -d "$CONFIG_DIR/sync.lock" ]
} }
# --- Neue Tests: Dry-Run-Verhalten ---
@test "Trockenlauf ist Standard ohne --execute" {
run_with_stubs env SSH_STUB_FAIL=0 DRY_RUN_SKIP_CONFIRM=1 bash "$SYNC_SCRIPT"
[ "$status" -eq 0 ]
[[ "$output" == *"TROCKENLAUF"* ]]
}
@test "Trockenlauf erstellt kein Backup" {
run_with_stubs env SSH_STUB_FAIL=0 DRY_RUN_SKIP_CONFIRM=1 bash "$SYNC_SCRIPT"
[ "$status" -eq 0 ]
[ ! -f "$HOME/.config/darktable/library.db.bak" ]
[ ! -f "$HOME/.config/darktable/data.db.bak" ]
}
@test "Trockenlauf loescht sync_pending nicht" {
touch "$CONFIG_DIR/sync_pending"
run_with_stubs env SSH_STUB_FAIL=0 DRY_RUN_SKIP_CONFIRM=1 bash "$SYNC_SCRIPT"
[ "$status" -eq 0 ]
[ -f "$CONFIG_DIR/sync_pending" ]
}
@test "--execute fuehrt echten Sync durch und erstellt Backup" {
run_with_stubs env SSH_STUB_FAIL=0 bash "$SYNC_SCRIPT" --execute
[ "$status" -eq 0 ]
[ -f "$HOME/.config/darktable/library.db.bak" ]
}
@test "-e ist Kurzform fuer --execute" {
run_with_stubs env SSH_STUB_FAIL=0 bash "$SYNC_SCRIPT" -e
[ "$status" -eq 0 ]
[ -f "$HOME/.config/darktable/library.db.bak" ]
}
@test "Trockenlauf zeigt Ergebnis-Zusammenfassung" {
run_with_stubs env SSH_STUB_FAIL=0 DRY_RUN_SKIP_CONFIRM=1 bash "$SYNC_SCRIPT"
[ "$status" -eq 0 ]
[[ "$output" == *"Trockenlauf-Ergebnis"* ]]
[[ "$output" == *"Upload:"* ]]
[[ "$output" == *"Download:"* ]]
}
@test "Trockenlauf zaehlt neue Dateien korrekt" {
run_with_stubs env SSH_STUB_FAIL=0 DRY_RUN_SKIP_CONFIRM=1 \
RSYNC_STUB_DRY_LINES=">f+++++++++ foto.jpg" bash "$SYNC_SCRIPT"
[ "$status" -eq 0 ]
[[ "$output" == *"neu"* ]]
}
+5
View File
@@ -23,6 +23,11 @@ SYNC_BIN=$HOME/.local/bin/darktable_sync.sh
EOF EOF
} }
# Raeumt nach jedem Test auf (verhindert Lock-Leakage zwischen Tests)
teardown() {
rm -rf "$CONFIG_DIR/sync.lock"
}
# Fuehrt ein Script mit dem Stubs-Verzeichnis vorne im PATH aus # Fuehrt ein Script mit dem Stubs-Verzeichnis vorne im PATH aus
run_with_stubs() { run_with_stubs() {
run env PATH="$STUBS_DIR:$PATH" "$@" run env PATH="$STUBS_DIR:$PATH" "$@"
+1 -1
View File
@@ -80,7 +80,7 @@ EOF
mkdir -p "$CONFIG_DIR/sync.lock" mkdir -p "$CONFIG_DIR/sync.lock"
run_with_stubs env SSH_STUB_FAIL=0 bash "$SYNC_SCRIPT" run_with_stubs env SSH_STUB_FAIL=0 bash "$SYNC_SCRIPT"
[ "$status" -eq 1 ] [ "$status" -eq 1 ]
[[ "$output" == *"laeuft bereits"* ]] [[ "$output" == *"läuft bereits"* ]]
rmdir "$CONFIG_DIR/sync.lock" rmdir "$CONFIG_DIR/sync.lock"
} }
+7
View File
@@ -1,7 +1,14 @@
#!/bin/bash #!/bin/bash
# rsync-Stub: Verhalten per Umgebungsvariable steuerbar # rsync-Stub: Verhalten per Umgebungsvariable steuerbar
# RSYNC_STUB_FAIL=1 → schlaegt fehl # RSYNC_STUB_FAIL=1 → schlaegt fehl
# RSYNC_STUB_DRY_LINES → Ausgabe bei --dry-run (Zeilenumbrüche als \n)
if [ "${RSYNC_STUB_FAIL:-0}" = "1" ]; then if [ "${RSYNC_STUB_FAIL:-0}" = "1" ]; then
exit 1 exit 1
fi fi
for arg in "$@"; do
if [ "$arg" = "--dry-run" ] && [ -n "${RSYNC_STUB_DRY_LINES:-}" ]; then
echo -e "$RSYNC_STUB_DRY_LINES"
break
fi
done
exit 0 exit 0