mirror of
https://github.com/Lakr233/vphone-cli.git
synced 2026-04-05 04:59:05 +08:00
* fix: prefer project venv Python for patchers * add: VM backup, restore, and switch support Named backups via rsync --sparse for efficient sparse disk handling. - vm_backup.sh: save current VM as a named backup to vm.backups/ - vm_restore.sh: restore a named backup into vm/ - vm_switch.sh: save current + restore target in one step - Makefile targets: vm_backup, vm_restore, vm_switch, vm_list - Documentation added to all READMEs (EN, ZH, KO, JA) Closes #204 Made-with: Cursor
119 lines
3.1 KiB
Bash
Executable File
119 lines
3.1 KiB
Bash
Executable File
#!/bin/zsh
|
|
# vm_switch.sh — Switch the active VM to a different named backup.
|
|
#
|
|
# Saves the current VM under its name (from vm/.vm_name), then restores
|
|
# the target backup. If the current VM has no name yet, prompts for one.
|
|
#
|
|
# Usage:
|
|
# make vm_switch NAME=ios18
|
|
set -euo pipefail
|
|
|
|
VM_DIR="${VM_DIR:-vm}"
|
|
BACKUPS_DIR="${BACKUPS_DIR:-vm.backups}"
|
|
NAME="${NAME:-}"
|
|
BACKUP_INCLUDE_IPSW="${BACKUP_INCLUDE_IPSW:-0}"
|
|
|
|
# --- Parse args ---
|
|
while [[ $# -gt 0 ]]; do
|
|
case "$1" in
|
|
--name) NAME="$2"; shift 2 ;;
|
|
--include-ipsw) BACKUP_INCLUDE_IPSW=1; shift ;;
|
|
-h|--help)
|
|
echo "Usage: $0 --name <target> [--include-ipsw]"
|
|
exit 0
|
|
;;
|
|
*) echo "Unknown option: $1"; exit 1 ;;
|
|
esac
|
|
done
|
|
|
|
if [[ -z "${NAME}" ]]; then
|
|
echo "ERROR: NAME is required (the backup to switch to)."
|
|
echo " Usage: make vm_switch NAME=ios18"
|
|
echo ""
|
|
echo "Available backups:"
|
|
if [[ -d "${BACKUPS_DIR}" ]]; then
|
|
for d in "${BACKUPS_DIR}"/*/; do
|
|
[[ -f "${d}config.plist" ]] && echo " - $(basename "${d}")"
|
|
done
|
|
else
|
|
echo " (none)"
|
|
fi
|
|
exit 1
|
|
fi
|
|
|
|
TARGET="${BACKUPS_DIR}/${NAME}"
|
|
|
|
if [[ ! -d "${TARGET}" || ! -f "${TARGET}/config.plist" ]]; then
|
|
echo "ERROR: Backup '${NAME}' not found."
|
|
echo ""
|
|
echo "Available backups:"
|
|
if [[ -d "${BACKUPS_DIR}" ]]; then
|
|
for d in "${BACKUPS_DIR}"/*/; do
|
|
[[ -f "${d}config.plist" ]] && echo " - $(basename "${d}")"
|
|
done
|
|
else
|
|
echo " (none)"
|
|
fi
|
|
exit 1
|
|
fi
|
|
|
|
# --- Check for running VM ---
|
|
if pgrep -f "vphone-cli.*--config.*${VM_DIR}" >/dev/null 2>&1; then
|
|
echo "ERROR: vphone-cli appears to be running against ${VM_DIR}."
|
|
echo " Stop the VM before switching."
|
|
exit 1
|
|
fi
|
|
|
|
# --- Determine current VM name ---
|
|
CURRENT=""
|
|
if [[ -d "${VM_DIR}" && -f "${VM_DIR}/config.plist" ]]; then
|
|
if [[ -f "${VM_DIR}/.vm_name" ]]; then
|
|
CURRENT="$(< "${VM_DIR}/.vm_name")"
|
|
fi
|
|
|
|
if [[ -z "${CURRENT}" ]]; then
|
|
echo "Current VM has no name. Give it one to save before switching."
|
|
printf "Name for current VM: "
|
|
read -r CURRENT
|
|
if [[ -z "${CURRENT}" ]]; then
|
|
echo "ERROR: Cannot switch without saving the current VM."
|
|
exit 1
|
|
fi
|
|
fi
|
|
|
|
if [[ "${CURRENT}" == "${NAME}" ]]; then
|
|
echo "'${NAME}' is already the active VM."
|
|
exit 0
|
|
fi
|
|
|
|
# --- Save current ---
|
|
echo "=== Saving current VM as '${CURRENT}' ==="
|
|
CURRENT_DEST="${BACKUPS_DIR}/${CURRENT}"
|
|
mkdir -p "${CURRENT_DEST}"
|
|
|
|
RSYNC_EXCLUDES=()
|
|
if [[ "${BACKUP_INCLUDE_IPSW}" != "1" ]]; then
|
|
RSYNC_EXCLUDES+=(--exclude '*_Restore*/')
|
|
fi
|
|
|
|
rsync -aH --sparse --progress --delete \
|
|
"${RSYNC_EXCLUDES[@]}" \
|
|
"${VM_DIR}/" "${CURRENT_DEST}/"
|
|
|
|
echo ""
|
|
fi
|
|
|
|
# --- Restore target ---
|
|
echo "=== Restoring '${NAME}' ==="
|
|
|
|
mkdir -p "${VM_DIR}"
|
|
|
|
rsync -aH --sparse --progress --delete \
|
|
"${TARGET}/" "${VM_DIR}/"
|
|
|
|
echo "${NAME}" > "${VM_DIR}/.vm_name"
|
|
|
|
echo ""
|
|
echo "=== Switched: ${CURRENT:+${CURRENT} → }${NAME} ==="
|
|
echo "Next: make boot"
|