fix(uninstall): MANDATORY tar backup in ALL modes (safe/settings-only/full)
Per user demand after data loss incident: never trust mode flags alone. Previously only `full` mode created tar.gz backup. Now `safe` and `settings-only` also create full tar.gz of ~/.claude (resp .codex, .gemini, .qwen) BEFORE any mutation. If backup fails, refuse to proceed for that user — skip to next. Restore is always one command: tar -xzf ~/.<tool>.uninstall-backup.<TS>.tar.gz -C ~ Applies to all 4 scripts: uclaude, ucodex, ugemini, uqwen. Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
This commit is contained in:
@@ -166,6 +166,24 @@ for user_home in /root /home/*; do
|
|||||||
CLAUDE_DIR="$user_home/.claude"
|
CLAUDE_DIR="$user_home/.claude"
|
||||||
[ -d "$CLAUDE_DIR" ] || continue
|
[ -d "$CLAUDE_DIR" ] || continue
|
||||||
|
|
||||||
|
# ALWAYS create full tar backup BEFORE any destructive operation, in EVERY mode.
|
||||||
|
# Per user demand after data loss incident: never trust mode flags alone.
|
||||||
|
# If backup fails, refuse to touch this user's data and skip to next user.
|
||||||
|
FULL_BACKUP_TAR="$user_home/.claude.uninstall-backup.${TIMESTAMP}.tar.gz"
|
||||||
|
info "Creating safety backup: $FULL_BACKUP_TAR ..."
|
||||||
|
if ! tar -czf "$FULL_BACKUP_TAR" -C "$user_home" .claude 2>/dev/null; then
|
||||||
|
err "Backup FAILED for $CLAUDE_DIR — REFUSING to proceed (data preserved)"
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
backup_size=$(du -h "$FULL_BACKUP_TAR" 2>/dev/null | awk '{print $1}')
|
||||||
|
log "Safety backup: $FULL_BACKUP_TAR ($backup_size)"
|
||||||
|
log "Restore anytime: tar -xzf $FULL_BACKUP_TAR -C $user_home"
|
||||||
|
|
||||||
|
# Also back up ~/.claude.json (onboarding state lives outside .claude/)
|
||||||
|
if [ -f "$user_home/.claude.json" ]; then
|
||||||
|
cp -p "$user_home/.claude.json" "$user_home/.claude.json.uninstall.bak.${TIMESTAMP}" 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
|
||||||
case "$MODE" in
|
case "$MODE" in
|
||||||
safe)
|
safe)
|
||||||
# Remove ONLY patcher-managed files. Preserve all user data.
|
# Remove ONLY patcher-managed files. Preserve all user data.
|
||||||
@@ -197,18 +215,9 @@ for user_home in /root /home/*; do
|
|||||||
;;
|
;;
|
||||||
|
|
||||||
full)
|
full)
|
||||||
# WIPE — but ALWAYS backup first.
|
# Backup already created above. Now wipe.
|
||||||
BACKUP_TAR="$user_home/.claude.uninstall-backup.${TIMESTAMP}.tar.gz"
|
rm -rf "$CLAUDE_DIR"
|
||||||
info "Creating backup: $BACKUP_TAR ..."
|
log "Removed $CLAUDE_DIR (full wipe)"
|
||||||
if tar -czf "$BACKUP_TAR" -C "$user_home" .claude 2>/dev/null; then
|
|
||||||
local_size=$(du -h "$BACKUP_TAR" 2>/dev/null | awk '{print $1}')
|
|
||||||
log "Backup saved: $BACKUP_TAR ($local_size)"
|
|
||||||
rm -rf "$CLAUDE_DIR"
|
|
||||||
log "Removed $CLAUDE_DIR"
|
|
||||||
log " Restore anytime: tar -xzf $BACKUP_TAR -C $user_home"
|
|
||||||
else
|
|
||||||
err "Backup FAILED — refusing to delete $CLAUDE_DIR (data preserved)"
|
|
||||||
fi
|
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
|
|||||||
@@ -113,8 +113,20 @@ fi
|
|||||||
|
|
||||||
# ---- Remove config ----
|
# ---- Remove config ----
|
||||||
|
|
||||||
for user_home in "${_home_dirs[@]}"; do CODEX_DIR="$user_home/.codex"
|
for user_home in "${_home_dirs[@]}"; do
|
||||||
|
CODEX_DIR="$user_home/.codex"
|
||||||
if [ -d "$CODEX_DIR" ]; then
|
if [ -d "$CODEX_DIR" ]; then
|
||||||
|
# ALWAYS create full tar backup BEFORE any destructive op, in EVERY mode.
|
||||||
|
# Per user demand after data loss incident: never trust mode flags alone.
|
||||||
|
FULL_BAK="$user_home/.codex.uninstall-backup.$TIMESTAMP.tar.gz"
|
||||||
|
echo " Creating safety backup: $FULL_BAK ..."
|
||||||
|
if ! tar -czf "$FULL_BAK" -C "$user_home" .codex 2>/dev/null; then
|
||||||
|
echo " ERROR: backup FAILED for $CODEX_DIR — REFUSING to proceed (data preserved)"
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
echo " Safety backup: $FULL_BAK ($(du -h "$FULL_BAK" 2>/dev/null | awk '{print $1}'))"
|
||||||
|
echo " Restore anytime: tar -xzf $FULL_BAK -C $user_home"
|
||||||
|
|
||||||
# PRESERVE user data by default. See uclaude_uninstall.sh history.
|
# PRESERVE user data by default. See uclaude_uninstall.sh history.
|
||||||
case "${MODE:-safe}" in
|
case "${MODE:-safe}" in
|
||||||
safe)
|
safe)
|
||||||
@@ -137,14 +149,8 @@ for user_home in "${_home_dirs[@]}"; do CODEX_DIR="$user_home/.codex"
|
|||||||
done
|
done
|
||||||
;;
|
;;
|
||||||
full)
|
full)
|
||||||
BAK="$user_home/.codex.uninstall-backup.$TIMESTAMP.tar.gz"
|
rm -rf "$user_home/.codex"
|
||||||
if tar -czf "$BAK" -C "$user_home" .codex 2>/dev/null; then
|
echo " Removed $user_home/.codex (full wipe)"
|
||||||
echo " Backup: $BAK"
|
|
||||||
rm -rf "$user_home/.codex"
|
|
||||||
echo " Removed $user_home/.codex (restore: tar -xzf $BAK -C $user_home)"
|
|
||||||
else
|
|
||||||
echo " ERROR: backup failed — refusing to delete $user_home/.codex"
|
|
||||||
fi
|
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
fi
|
fi
|
||||||
|
|||||||
@@ -103,8 +103,20 @@ fi
|
|||||||
|
|
||||||
# ---- Remove settings ----
|
# ---- Remove settings ----
|
||||||
|
|
||||||
for user_home in /root $SCAN_DIRS/*; do GEMINI_DIR="$user_home/.gemini"
|
for user_home in /root $SCAN_DIRS/*; do
|
||||||
|
GEMINI_DIR="$user_home/.gemini"
|
||||||
if [ -d "$GEMINI_DIR" ]; then
|
if [ -d "$GEMINI_DIR" ]; then
|
||||||
|
# ALWAYS create full tar backup BEFORE any destructive op, in EVERY mode.
|
||||||
|
# Per user demand after data loss incident: never trust mode flags alone.
|
||||||
|
FULL_BAK="$user_home/.gemini.uninstall-backup.$TIMESTAMP.tar.gz"
|
||||||
|
echo " Creating safety backup: $FULL_BAK ..."
|
||||||
|
if ! tar -czf "$FULL_BAK" -C "$user_home" .gemini 2>/dev/null; then
|
||||||
|
echo " ERROR: backup FAILED for $GEMINI_DIR — REFUSING to proceed (data preserved)"
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
echo " Safety backup: $FULL_BAK ($(du -h "$FULL_BAK" 2>/dev/null | awk '{print $1}'))"
|
||||||
|
echo " Restore anytime: tar -xzf $FULL_BAK -C $user_home"
|
||||||
|
|
||||||
# PRESERVE user data by default. See uclaude_uninstall.sh history.
|
# PRESERVE user data by default. See uclaude_uninstall.sh history.
|
||||||
case "${MODE:-safe}" in
|
case "${MODE:-safe}" in
|
||||||
safe)
|
safe)
|
||||||
@@ -127,14 +139,8 @@ for user_home in /root $SCAN_DIRS/*; do GEMINI_DIR="$user_home/.gemini"
|
|||||||
done
|
done
|
||||||
;;
|
;;
|
||||||
full)
|
full)
|
||||||
BAK="$user_home/.gemini.uninstall-backup.$TIMESTAMP.tar.gz"
|
rm -rf "$user_home/.gemini"
|
||||||
if tar -czf "$BAK" -C "$user_home" .gemini 2>/dev/null; then
|
echo " Removed $user_home/.gemini (full wipe)"
|
||||||
echo " Backup: $BAK"
|
|
||||||
rm -rf "$user_home/.gemini"
|
|
||||||
echo " Removed $user_home/.gemini (restore: tar -xzf $BAK -C $user_home)"
|
|
||||||
else
|
|
||||||
echo " ERROR: backup failed — refusing to delete $user_home/.gemini"
|
|
||||||
fi
|
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
fi
|
fi
|
||||||
|
|||||||
@@ -103,8 +103,20 @@ fi
|
|||||||
|
|
||||||
# ---- Remove settings ----
|
# ---- Remove settings ----
|
||||||
|
|
||||||
for user_home in /root $SCAN_DIRS/*; do QWEN_DIR="$user_home/.qwen"
|
for user_home in /root $SCAN_DIRS/*; do
|
||||||
|
QWEN_DIR="$user_home/.qwen"
|
||||||
if [ -d "$QWEN_DIR" ]; then
|
if [ -d "$QWEN_DIR" ]; then
|
||||||
|
# ALWAYS create full tar backup BEFORE any destructive op, in EVERY mode.
|
||||||
|
# Per user demand after data loss incident: never trust mode flags alone.
|
||||||
|
FULL_BAK="$user_home/.qwen.uninstall-backup.$TIMESTAMP.tar.gz"
|
||||||
|
echo " Creating safety backup: $FULL_BAK ..."
|
||||||
|
if ! tar -czf "$FULL_BAK" -C "$user_home" .qwen 2>/dev/null; then
|
||||||
|
echo " ERROR: backup FAILED for $QWEN_DIR — REFUSING to proceed (data preserved)"
|
||||||
|
continue
|
||||||
|
fi
|
||||||
|
echo " Safety backup: $FULL_BAK ($(du -h "$FULL_BAK" 2>/dev/null | awk '{print $1}'))"
|
||||||
|
echo " Restore anytime: tar -xzf $FULL_BAK -C $user_home"
|
||||||
|
|
||||||
# PRESERVE user data by default. See uclaude_uninstall.sh history.
|
# PRESERVE user data by default. See uclaude_uninstall.sh history.
|
||||||
case "${MODE:-safe}" in
|
case "${MODE:-safe}" in
|
||||||
safe)
|
safe)
|
||||||
@@ -127,14 +139,8 @@ for user_home in /root $SCAN_DIRS/*; do QWEN_DIR="$user_home/.qwen"
|
|||||||
done
|
done
|
||||||
;;
|
;;
|
||||||
full)
|
full)
|
||||||
BAK="$user_home/.qwen.uninstall-backup.$TIMESTAMP.tar.gz"
|
rm -rf "$user_home/.qwen"
|
||||||
if tar -czf "$BAK" -C "$user_home" .qwen 2>/dev/null; then
|
echo " Removed $user_home/.qwen (full wipe)"
|
||||||
echo " Backup: $BAK"
|
|
||||||
rm -rf "$user_home/.qwen"
|
|
||||||
echo " Removed $user_home/.qwen (restore: tar -xzf $BAK -C $user_home)"
|
|
||||||
else
|
|
||||||
echo " ERROR: backup failed — refusing to delete $user_home/.qwen"
|
|
||||||
fi
|
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
fi
|
fi
|
||||||
|
|||||||
Reference in New Issue
Block a user