Files
clawgo/install.sh
2026-03-10 14:18:29 +08:00

417 lines
10 KiB
Bash
Executable File

#!/usr/bin/env bash
set -euo pipefail
OWNER="YspCoder"
REPO="clawgo"
BIN="clawgo"
INSTALL_DIR="/usr/local/bin"
VARIANT="${CLAWGO_CHANNEL_VARIANT:-full}"
VARIANT_EXPLICIT=0
CHANNEL_VARIANTS=(full none telegram discord feishu maixcam qq dingtalk whatsapp)
CONFIG_DIR="$HOME/.clawgo"
CONFIG_PATH="$CONFIG_DIR/config.json"
WORKSPACE_DIR="$HOME/.clawgo/workspace"
LEGACY_WORKSPACE_DIR="$HOME/.openclaw/workspace"
usage() {
cat <<EOF
Usage: $0 [--variant full|none|telegram|discord|feishu|maixcam|qq|dingtalk|whatsapp]
Install or upgrade ClawGo from the latest GitHub release.
Notes:
- WebUI is embedded in the binary and initialized when you run 'clawgo onboard'.
- Variant 'none' installs the no-channel build.
- OpenClaw migration is offered only when a legacy workspace is detected.
EOF
}
log() {
printf '%s\n' "$*"
}
warn() {
printf 'Warning: %s\n' "$*" >&2
}
require_cmd() {
if ! command -v "$1" >/dev/null 2>&1; then
warn "Missing required command: $1"
exit 1
fi
}
prompt_yes_no() {
local prompt="$1"
local default="${2:-N}"
local reply
if [[ ! -r /dev/tty ]]; then
[[ "$default" =~ ^[Yy]$ ]]
return
fi
if [[ "$default" =~ ^[Yy]$ ]]; then
read -r -p "$prompt [Y/n]: " reply < /dev/tty || true
reply="${reply:-Y}"
else
read -r -p "$prompt [y/N]: " reply < /dev/tty || true
reply="${reply:-N}"
fi
[[ "$reply" =~ ^[Yy]$ ]]
}
tty_read() {
local __var_name="$1"
local __prompt="$2"
local __default="${3:-}"
local __silent="${4:-0}"
local __reply=""
if [[ ! -r /dev/tty ]]; then
printf -v "$__var_name" '%s' "$__default"
return
fi
if [[ "$__silent" == "1" ]]; then
read -r -s -p "$__prompt" __reply < /dev/tty || true
echo > /dev/tty
else
read -r -p "$__prompt" __reply < /dev/tty || true
fi
__reply="${__reply:-$__default}"
printf -v "$__var_name" '%s' "$__reply"
}
is_valid_variant() {
local candidate="$1"
local item
for item in "${CHANNEL_VARIANTS[@]}"; do
if [[ "$item" == "$candidate" ]]; then
return 0
fi
done
return 1
}
parse_args() {
while [[ $# -gt 0 ]]; do
case "$1" in
--variant)
if [[ $# -lt 2 ]]; then
warn "--variant requires a value"
exit 1
fi
VARIANT="$2"
VARIANT_EXPLICIT=1
shift 2
;;
-h|--help)
usage
exit 0
;;
*)
warn "Unknown argument: $1"
usage
exit 1
;;
esac
done
}
choose_variant() {
if ! is_valid_variant "$VARIANT"; then
warn "Unsupported variant: $VARIANT"
exit 1
fi
if [[ "$VARIANT_EXPLICIT" == "1" ]]; then
log "Selected variant: $VARIANT"
return
fi
if [[ ! -r /dev/tty ]]; then
log "Selected variant: $VARIANT"
return
fi
log "Choose install variant:"
log " 1. full Full build with all channels"
log " 2. none No-channel build"
log " 3. telegram Telegram-only build"
log " 4. discord Discord-only build"
log " 5. feishu Feishu-only build"
log " 6. maixcam MaixCam-only build"
log " 7. qq QQ-only build"
log " 8. dingtalk DingTalk-only build"
log " 9. whatsapp WhatsApp-only build"
local choice default_choice
case "$VARIANT" in
full) default_choice="1" ;;
none) default_choice="2" ;;
telegram) default_choice="3" ;;
discord) default_choice="4" ;;
feishu) default_choice="5" ;;
maixcam) default_choice="6" ;;
qq) default_choice="7" ;;
dingtalk) default_choice="8" ;;
whatsapp) default_choice="9" ;;
*) default_choice="1" ;;
esac
tty_read choice "Enter your choice (1-9, default $default_choice): " "$default_choice"
case "$choice" in
1) VARIANT="full" ;;
2) VARIANT="none" ;;
3) VARIANT="telegram" ;;
4) VARIANT="discord" ;;
5) VARIANT="feishu" ;;
6) VARIANT="maixcam" ;;
7) VARIANT="qq" ;;
8) VARIANT="dingtalk" ;;
9) VARIANT="whatsapp" ;;
*)
warn "Invalid variant selection: $choice"
exit 1
;;
esac
log "Selected variant: $VARIANT"
}
detect_platform() {
OS="$(uname | tr '[:upper:]' '[:lower:]')"
ARCH="$(uname -m)"
case "$ARCH" in
x86_64) ARCH="amd64" ;;
aarch64|arm64) ARCH="arm64" ;;
*)
warn "Unsupported architecture: $ARCH"
exit 1
;;
esac
case "$OS" in
linux|darwin) ;;
*)
warn "Unsupported operating system: $OS"
exit 1
;;
esac
log "Detected OS=$OS ARCH=$ARCH"
}
fetch_latest_tag() {
require_cmd curl
local api="https://api.github.com/repos/$OWNER/$REPO/releases/latest"
TAG="$(curl -fsSL "$api" | sed -n 's/.*"tag_name": *"\([^"]*\)".*/\1/p' | head -n1)"
if [[ -z "${TAG:-}" ]]; then
warn "Unable to get latest release tag from GitHub"
exit 1
fi
log "Latest Release: $TAG"
}
install_binary() {
local suffix=""
if [[ "$VARIANT" == "none" ]]; then
suffix="-nochannels"
elif [[ "$VARIANT" != "full" ]]; then
suffix="-$VARIANT"
fi
local file="${BIN}-${OS}-${ARCH}${suffix}.tar.gz"
local url="https://github.com/$OWNER/$REPO/releases/download/$TAG/$file"
local out="$TMPDIR/$file"
log "Downloading $file..."
curl -fSL "$url" -o "$out"
tar -xzf "$out" -C "$TMPDIR"
local extracted_bin=""
if [[ -f "$TMPDIR/$BIN" ]]; then
extracted_bin="$TMPDIR/$BIN"
elif [[ -f "$TMPDIR/${BIN}-${OS}-${ARCH}${suffix}" ]]; then
extracted_bin="$TMPDIR/${BIN}-${OS}-${ARCH}${suffix}"
else
extracted_bin="$(find "$TMPDIR" -maxdepth 3 -type f \( -name "$BIN" -o -name "${BIN}-${OS}-${ARCH}${suffix}" -o -name "${BIN}-*" \) ! -name "*.tar.gz" ! -name "*.zip" | head -n1)"
fi
if [[ -z "$extracted_bin" || ! -f "$extracted_bin" ]]; then
warn "Failed to locate extracted binary from $file"
exit 1
fi
chmod +x "$extracted_bin"
if command -v "$BIN" >/dev/null 2>&1 || [[ -f "$INSTALL_DIR/$BIN" ]]; then
log "Updating existing $BIN in $INSTALL_DIR ..."
else
log "Installing $BIN to $INSTALL_DIR ..."
fi
sudo mv "$extracted_bin" "$INSTALL_DIR/$BIN"
log "Installed $BIN to $INSTALL_DIR/$BIN"
}
migrate_local_openclaw() {
local src="${1:-$LEGACY_WORKSPACE_DIR}"
local dst="${2:-$WORKSPACE_DIR}"
if [[ ! -d "$src" ]]; then
warn "OpenClaw workspace not found: $src"
return 1
fi
log "[INFO] source: $src"
log "[INFO] target: $dst"
mkdir -p "$dst" "$dst/memory"
local ts
ts="$(date -u +%Y%m%dT%H%M%SZ)"
local backup_dir="$dst/.migration-backup-$ts"
mkdir -p "$backup_dir"
for f in AGENTS.md SOUL.md USER.md IDENTITY.md TOOLS.md MEMORY.md HEARTBEAT.md; do
if [[ -f "$dst/$f" ]]; then
cp -a "$dst/$f" "$backup_dir/$f"
fi
done
if [[ -d "$dst/memory" ]]; then
cp -a "$dst/memory" "$backup_dir/memory" || true
fi
for f in AGENTS.md SOUL.md USER.md IDENTITY.md TOOLS.md MEMORY.md HEARTBEAT.md; do
if [[ -f "$src/$f" ]]; then
cp -a "$src/$f" "$dst/$f"
log "[OK] migrated $f"
fi
done
if [[ -d "$src/memory" ]]; then
rsync -a "$src/memory/" "$dst/memory/"
log "[OK] migrated memory/"
fi
log "[DONE] local migration complete"
}
migrate_remote_openclaw() {
require_cmd sshpass
require_cmd scp
require_cmd ssh
local remote_host remote_port remote_pass migration_script
tty_read remote_host "Enter remote host (e.g. user@hostname): "
tty_read remote_port "Enter remote port (default 22): " "22"
tty_read remote_pass "Enter remote password: " "" "1"
migration_script="$TMPDIR/openclaw2clawgo.sh"
cat > "$migration_script" <<'EOF'
#!/usr/bin/env bash
set -euo pipefail
SRC_DEFAULT="$HOME/.openclaw/workspace"
DST_DEFAULT="$HOME/.clawgo/workspace"
SRC="${1:-$SRC_DEFAULT}"
DST="${2:-$DST_DEFAULT}"
mkdir -p "$DST" "$DST/memory"
TS="$(date -u +%Y%m%dT%H%M%SZ)"
BACKUP_DIR="$DST/.migration-backup-$TS"
mkdir -p "$BACKUP_DIR"
for f in AGENTS.md SOUL.md USER.md IDENTITY.md TOOLS.md MEMORY.md HEARTBEAT.md; do
if [[ -f "$DST/$f" ]]; then
cp -a "$DST/$f" "$BACKUP_DIR/$f"
fi
done
if [[ -d "$DST/memory" ]]; then
cp -a "$DST/memory" "$BACKUP_DIR/memory" || true
fi
for f in AGENTS.md SOUL.md USER.md IDENTITY.md TOOLS.md MEMORY.md HEARTBEAT.md; do
if [[ -f "$SRC/$f" ]]; then
cp -a "$SRC/$f" "$DST/$f"
echo "[OK] migrated $f"
fi
done
if [[ -d "$SRC/memory" ]]; then
rsync -a "$SRC/memory/" "$DST/memory/"
echo "[OK] migrated memory/"
fi
echo "[DONE] remote migration complete"
EOF
chmod +x "$migration_script"
sshpass -p "$remote_pass" scp -P "$remote_port" "$migration_script" "$remote_host:/tmp/openclaw2clawgo.sh"
sshpass -p "$remote_pass" ssh -p "$remote_port" "$remote_host" "bash /tmp/openclaw2clawgo.sh"
log "[INFO] Remote migration completed."
}
offer_openclaw_migration() {
if [[ ! -d "$LEGACY_WORKSPACE_DIR" ]]; then
return
fi
if ! prompt_yes_no "Detected OpenClaw workspace at $LEGACY_WORKSPACE_DIR. Migrate it to ClawGo now?" "N"; then
return
fi
local migration_type
if [[ -r /dev/tty ]]; then
log "Choose migration type:"
log " 1. Local migration"
log " 2. Remote migration"
tty_read migration_type "Enter your choice (1 or 2): " "1"
else
migration_type="1"
fi
case "$migration_type" in
1)
warn "Migration will overwrite files in $WORKSPACE_DIR when names collide."
if prompt_yes_no "Continue local migration?" "N"; then
migrate_local_openclaw
else
log "Migration skipped."
fi
;;
2)
migrate_remote_openclaw
;;
*)
log "Invalid choice. Skipping migration."
;;
esac
}
offer_onboard() {
log "Refreshing embedded WebUI assets..."
"$INSTALL_DIR/$BIN" onboard --sync-webui >/dev/null 2>&1 || warn "Failed to refresh embedded WebUI assets automatically. You can run 'clawgo onboard --sync-webui' later."
if [[ -f "$CONFIG_PATH" ]]; then
log "Existing config detected at $CONFIG_PATH"
log "WebUI assets were refreshed from the embedded bundle."
log "Run 'clawgo onboard' only if you want to regenerate config or missing workspace templates."
return
fi
log "WebUI assets were refreshed from the embedded bundle."
if prompt_yes_no "No config found. Run 'clawgo onboard' now?" "N"; then
"$INSTALL_DIR/$BIN" onboard
else
log "Skipped onboard. Run 'clawgo onboard' when you are ready."
fi
}
main() {
parse_args "$@"
detect_platform
fetch_latest_tag
choose_variant
TMPDIR="$(mktemp -d)"
trap 'rm -rf "$TMPDIR"' EXIT
install_binary
offer_openclaw_migration
offer_onboard
log "Done."
log "Run 'clawgo --help' to verify the installation."
}
main "$@"