You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
3317 lines
122 KiB
3317 lines
122 KiB
#!/bin/bash
|
|
|
|
# ========================================
|
|
# Cursor macOS 机器码修改脚本
|
|
# ========================================
|
|
#
|
|
# 🔧 权限修复增强:
|
|
# - 集成用户提供的核心权限修复命令
|
|
# - 特别处理logs目录权限问题
|
|
# - 解决EACCES: permission denied错误
|
|
# - 确保Cursor能正常启动
|
|
#
|
|
# 🚨 如果遇到权限错误,脚本会自动执行:
|
|
# - sudo chown -R "$TARGET_USER" "$TARGET_HOME/Library/Application Support/Cursor"
|
|
# - sudo chown -R "$TARGET_USER" "$TARGET_HOME/.cursor"
|
|
# - chmod -R u+rwX "$TARGET_HOME/Library/Application Support/Cursor"
|
|
# - chmod -R u+rwX "$TARGET_HOME/.cursor"
|
|
#
|
|
# ========================================
|
|
|
|
# 设置错误处理
|
|
set -e
|
|
|
|
# 定义日志文件路径
|
|
LOG_FILE="/tmp/cursor_free_trial_reset.log"
|
|
|
|
# 初始化日志文件
|
|
initialize_log() {
|
|
echo "========== Cursor Free Trial Reset Tool Log Start $(date) ==========" > "$LOG_FILE"
|
|
chmod 644 "$LOG_FILE"
|
|
}
|
|
|
|
# 颜色定义
|
|
RED='\033[0;31m'
|
|
GREEN='\033[0;32m'
|
|
YELLOW='\033[1;33m'
|
|
BLUE='\033[0;34m'
|
|
NC='\033[0m' # No Color
|
|
|
|
# 启动时尝试调整终端窗口大小为 120x40(列x行);不支持/失败时静默忽略,避免影响脚本主流程
|
|
try_resize_terminal_window() {
|
|
local target_cols=120
|
|
local target_rows=40
|
|
|
|
# 仅在交互终端中尝试,避免输出被重定向时出现乱码
|
|
if [ ! -t 1 ]; then
|
|
return 0
|
|
fi
|
|
|
|
case "${TERM:-}" in
|
|
""|dumb)
|
|
return 0
|
|
;;
|
|
esac
|
|
|
|
# 终端类型检测:仅对常见 xterm 体系终端尝试窗口调整(Terminal.app/iTerm2 以及常见终端通常为 xterm*)
|
|
case "${TERM:-}" in
|
|
xterm*|screen*|tmux*|rxvt*|alacritty*|kitty*|foot*|wezterm*)
|
|
;;
|
|
*)
|
|
return 0
|
|
;;
|
|
esac
|
|
|
|
# 优先通过 xterm 窗口控制序列调整;在 tmux/screen 下需要 passthrough 包装
|
|
if [ -n "${TMUX:-}" ]; then
|
|
printf '\033Ptmux;\033\033[8;%d;%dt\033\\' "$target_rows" "$target_cols" 2>/dev/null || true
|
|
elif [ -n "${STY:-}" ]; then
|
|
printf '\033P\033[8;%d;%dt\033\\' "$target_rows" "$target_cols" 2>/dev/null || true
|
|
else
|
|
printf '\033[8;%d;%dt' "$target_rows" "$target_cols" 2>/dev/null || true
|
|
fi
|
|
|
|
return 0
|
|
}
|
|
|
|
# 日志函数 - 同时输出到终端和日志文件
|
|
log_info() {
|
|
echo -e "${GREEN}[INFO]${NC} $1"
|
|
echo "[INFO] $(date '+%Y-%m-%d %H:%M:%S') $1" >> "$LOG_FILE"
|
|
}
|
|
|
|
log_warn() {
|
|
echo -e "${YELLOW}[WARN]${NC} $1"
|
|
echo "[WARN] $(date '+%Y-%m-%d %H:%M:%S') $1" >> "$LOG_FILE"
|
|
}
|
|
|
|
log_error() {
|
|
echo -e "${RED}[ERROR]${NC} $1"
|
|
echo "[ERROR] $(date '+%Y-%m-%d %H:%M:%S') $1" >> "$LOG_FILE"
|
|
}
|
|
|
|
log_debug() {
|
|
echo -e "${BLUE}[DEBUG]${NC} $1"
|
|
echo "[DEBUG] $(date '+%Y-%m-%d %H:%M:%S') $1" >> "$LOG_FILE"
|
|
}
|
|
|
|
# 记录命令输出到日志文件
|
|
log_cmd_output() {
|
|
local cmd="$1"
|
|
local msg="$2"
|
|
echo "[CMD] $(date '+%Y-%m-%d %H:%M:%S') 执行命令: $cmd" >> "$LOG_FILE"
|
|
echo "[CMD] $msg:" >> "$LOG_FILE"
|
|
eval "$cmd" 2>&1 | tee -a "$LOG_FILE"
|
|
echo "" >> "$LOG_FILE"
|
|
}
|
|
|
|
# 生成指定字节长度的十六进制串(2*bytes 个字符),优先 openssl,缺失时使用 python3 兜底
|
|
generate_hex_bytes() {
|
|
local bytes="$1"
|
|
if command -v openssl >/dev/null 2>&1; then
|
|
openssl rand -hex "$bytes"
|
|
return 0
|
|
fi
|
|
# mac 脚本已要求 python3,可作为兜底
|
|
python3 -c 'import os, sys; print(os.urandom(int(sys.argv[1])).hex())' "$bytes"
|
|
}
|
|
|
|
# 生成随机 UUID(小写),优先 uuidgen,缺失时使用 python3 兜底
|
|
generate_uuid() {
|
|
if command -v uuidgen >/dev/null 2>&1; then
|
|
uuidgen | tr '[:upper:]' '[:lower:]'
|
|
return 0
|
|
fi
|
|
# mac 脚本已要求 python3,可作为兜底
|
|
python3 -c 'import uuid; print(str(uuid.uuid4()))'
|
|
}
|
|
|
|
# 🚀 新增 Cursor 防掉试用Pro删除文件夹功能
|
|
remove_cursor_trial_folders() {
|
|
echo
|
|
log_info "🎯 [核心功能] 正在执行 Cursor 防掉试用Pro删除文件夹..."
|
|
log_info "📋 [说明] 此功能将删除指定的Cursor相关文件夹以重置试用状态"
|
|
echo
|
|
|
|
# 定义需要删除的文件夹路径
|
|
local folders_to_delete=(
|
|
"$TARGET_HOME/Library/Application Support/Cursor"
|
|
"$TARGET_HOME/.cursor"
|
|
)
|
|
|
|
log_info "📂 [检测] 将检查以下文件夹:"
|
|
for folder in "${folders_to_delete[@]}"; do
|
|
echo " 📁 $folder"
|
|
done
|
|
echo
|
|
|
|
local deleted_count=0
|
|
local skipped_count=0
|
|
local error_count=0
|
|
|
|
# 删除指定文件夹
|
|
for folder in "${folders_to_delete[@]}"; do
|
|
log_debug "🔍 [检查] 检查文件夹: $folder"
|
|
|
|
if [ -d "$folder" ]; then
|
|
log_warn "⚠️ [警告] 发现文件夹存在,正在删除..."
|
|
if rm -rf "$folder"; then
|
|
log_info "✅ [成功] 已删除文件夹: $folder"
|
|
((deleted_count++))
|
|
else
|
|
log_error "❌ [错误] 删除文件夹失败: $folder"
|
|
((error_count++))
|
|
fi
|
|
else
|
|
log_warn "⏭️ [跳过] 文件夹不存在: $folder"
|
|
((skipped_count++))
|
|
fi
|
|
echo
|
|
done
|
|
|
|
# 🔧 重要:删除文件夹后立即执行权限修复
|
|
log_info "� [权限修复] 删除文件夹后立即执行权限修复..."
|
|
echo
|
|
|
|
# 调用统一的权限修复函数
|
|
ensure_cursor_directory_permissions
|
|
|
|
# 显示操作统计
|
|
log_info "📊 [统计] 操作完成统计:"
|
|
echo " ✅ 成功删除: $deleted_count 个文件夹"
|
|
echo " ⏭️ 跳过处理: $skipped_count 个文件夹"
|
|
echo " ❌ 删除失败: $error_count 个文件夹"
|
|
echo
|
|
|
|
if [ $deleted_count -gt 0 ]; then
|
|
log_info "🎉 [完成] Cursor 防掉试用Pro文件夹删除完成!"
|
|
else
|
|
log_warn "🤔 [提示] 未找到需要删除的文件夹,可能已经清理过了"
|
|
fi
|
|
echo
|
|
}
|
|
|
|
# 🔄 重启Cursor并等待配置文件生成
|
|
restart_cursor_and_wait() {
|
|
echo
|
|
log_info "🔄 [重启] 正在重启Cursor以重新生成配置文件..."
|
|
|
|
if [ -z "$CURSOR_PROCESS_PATH" ]; then
|
|
log_error "❌ [错误] 未找到Cursor进程信息,无法重启"
|
|
return 1
|
|
fi
|
|
|
|
log_info "📍 [路径] 使用路径: $CURSOR_PROCESS_PATH"
|
|
|
|
if [ ! -f "$CURSOR_PROCESS_PATH" ]; then
|
|
log_error "❌ [错误] Cursor可执行文件不存在: $CURSOR_PROCESS_PATH"
|
|
return 1
|
|
fi
|
|
|
|
# 🔧 启动前权限修复
|
|
log_info "🔧 [启动前权限] 执行启动前权限修复..."
|
|
ensure_cursor_directory_permissions
|
|
|
|
# 启动Cursor
|
|
log_info "🚀 [启动] 正在启动Cursor..."
|
|
"$CURSOR_PROCESS_PATH" > /dev/null 2>&1 &
|
|
CURSOR_PID=$!
|
|
|
|
log_info "⏳ [等待] 等待15秒让Cursor完全启动并生成配置文件..."
|
|
sleep 15
|
|
|
|
# 检查配置文件是否生成
|
|
local config_path="$TARGET_HOME/Library/Application Support/Cursor/User/globalStorage/storage.json"
|
|
local max_wait=30
|
|
local waited=0
|
|
|
|
while [ ! -f "$config_path" ] && [ $waited -lt $max_wait ]; do
|
|
log_info "⏳ [等待] 等待配置文件生成... ($waited/$max_wait 秒)"
|
|
sleep 1
|
|
waited=$((waited + 1))
|
|
done
|
|
|
|
if [ -f "$config_path" ]; then
|
|
log_info "✅ [成功] 配置文件已生成: $config_path"
|
|
|
|
# 🛡️ 关键修复:配置文件生成后立即确保权限正确
|
|
ensure_cursor_directory_permissions
|
|
else
|
|
log_warn "⚠️ [警告] 配置文件未在预期时间内生成,继续执行..."
|
|
|
|
# 即使配置文件未生成,也要确保目录权限正确
|
|
ensure_cursor_directory_permissions
|
|
fi
|
|
|
|
# 强制关闭Cursor
|
|
log_info "🔄 [关闭] 正在关闭Cursor以进行配置修改..."
|
|
if [ ! -z "$CURSOR_PID" ]; then
|
|
kill $CURSOR_PID 2>/dev/null || true
|
|
fi
|
|
|
|
# 确保所有Cursor进程都关闭
|
|
pkill -f "Cursor" 2>/dev/null || true
|
|
|
|
log_info "✅ [完成] Cursor重启流程完成"
|
|
return 0
|
|
}
|
|
|
|
# 🔍 检查Cursor环境
|
|
test_cursor_environment() {
|
|
local mode=${1:-"FULL"}
|
|
|
|
echo
|
|
log_info "🔍 [环境检查] 正在检查Cursor环境..."
|
|
|
|
local config_path="$TARGET_HOME/Library/Application Support/Cursor/User/globalStorage/storage.json"
|
|
local cursor_app_data="$TARGET_HOME/Library/Application Support/Cursor"
|
|
local cursor_app_path="/Applications/Cursor.app"
|
|
local issues=()
|
|
|
|
# 检查Python3环境(macOS版本需要)
|
|
if ! command -v python3 >/dev/null 2>&1; then
|
|
issues+=("Python3环境不可用,macOS版本需要Python3来处理JSON配置文件")
|
|
log_warn "⚠️ [警告] 未找到Python3,请安装Python3: brew install python3"
|
|
else
|
|
log_info "✅ [检查] Python3环境可用: $(python3 --version)"
|
|
fi
|
|
|
|
# 检查配置文件
|
|
if [ ! -f "$config_path" ]; then
|
|
issues+=("配置文件不存在: $config_path")
|
|
else
|
|
# 验证JSON格式
|
|
# 🔧 修复:避免把路径直接拼进 Python 源码字符串(路径包含引号等特殊字符时会导致语法错误)
|
|
if python3 -c 'import json, sys; json.load(open(sys.argv[1], "r", encoding="utf-8"))' "$config_path" 2>/dev/null; then
|
|
log_info "✅ [检查] 配置文件格式正确"
|
|
else
|
|
issues+=("配置文件格式错误或损坏")
|
|
fi
|
|
fi
|
|
|
|
# 检查Cursor目录结构
|
|
if [ ! -d "$cursor_app_data" ]; then
|
|
issues+=("Cursor应用数据目录不存在: $cursor_app_data")
|
|
fi
|
|
|
|
# 检查Cursor应用安装
|
|
if [ ! -d "$cursor_app_path" ]; then
|
|
issues+=("未找到Cursor应用安装: $cursor_app_path")
|
|
else
|
|
log_info "✅ [检查] 找到Cursor应用: $cursor_app_path"
|
|
fi
|
|
|
|
# 检查目录权限
|
|
if [ -d "$cursor_app_data" ] && [ ! -w "$cursor_app_data" ]; then
|
|
issues+=("Cursor应用数据目录无写入权限: $cursor_app_data")
|
|
fi
|
|
|
|
# 返回检查结果
|
|
if [ ${#issues[@]} -eq 0 ]; then
|
|
log_info "✅ [环境检查] 所有检查通过"
|
|
return 0
|
|
else
|
|
log_error "❌ [环境检查] 发现 ${#issues[@]} 个问题:"
|
|
for issue in "${issues[@]}"; do
|
|
echo -e "${RED} • $issue${NC}"
|
|
done
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 🚀 启动Cursor生成配置文件
|
|
start_cursor_to_generate_config() {
|
|
log_info "🚀 [启动] 正在尝试启动Cursor生成配置文件..."
|
|
|
|
local cursor_app_path="/Applications/Cursor.app"
|
|
local cursor_executable="$cursor_app_path/Contents/MacOS/Cursor"
|
|
|
|
if [ ! -f "$cursor_executable" ]; then
|
|
log_error "❌ [错误] 未找到Cursor可执行文件: $cursor_executable"
|
|
return 1
|
|
fi
|
|
|
|
log_info "📍 [路径] 使用Cursor路径: $cursor_executable"
|
|
|
|
# 🚀 启动前权限修复
|
|
ensure_cursor_directory_permissions
|
|
|
|
# 启动Cursor
|
|
"$cursor_executable" > /dev/null 2>&1 &
|
|
local cursor_pid=$!
|
|
log_info "🚀 [启动] Cursor已启动,PID: $cursor_pid"
|
|
|
|
log_info "⏳ [等待] 请等待Cursor完全加载(约30秒)..."
|
|
log_info "💡 [提示] 您可以在Cursor完全加载后手动关闭它"
|
|
|
|
# 等待配置文件生成
|
|
local config_path="$TARGET_HOME/Library/Application Support/Cursor/User/globalStorage/storage.json"
|
|
local max_wait=60
|
|
local waited=0
|
|
|
|
while [ ! -f "$config_path" ] && [ $waited -lt $max_wait ]; do
|
|
sleep 2
|
|
waited=$((waited + 2))
|
|
if [ $((waited % 10)) -eq 0 ]; then
|
|
log_info "⏳ [等待] 等待配置文件生成... ($waited/$max_wait 秒)"
|
|
fi
|
|
done
|
|
|
|
if [ -f "$config_path" ]; then
|
|
log_info "✅ [成功] 配置文件已生成!"
|
|
log_info "💡 [提示] 现在可以关闭Cursor并重新运行脚本"
|
|
return 0
|
|
else
|
|
log_warn "⚠️ [超时] 配置文件未在预期时间内生成"
|
|
log_info "💡 [建议] 请手动操作Cursor(如创建新文件)以触发配置生成"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 🛡️ 统一权限修复函数(优化版本)
|
|
ensure_cursor_directory_permissions() {
|
|
log_info "🛡️ [权限修复] 执行核心权限修复命令..."
|
|
|
|
# ⚠️ 关键:不要用 $(whoami) 当作目标用户!在 sudo 场景下 whoami= root,会把用户目录 chown 成 root,导致 Cursor 启动 EACCES
|
|
local target_user="${TARGET_USER:-${SUDO_USER:-$USER}}"
|
|
local cursor_support_dir="$TARGET_HOME/Library/Application Support/Cursor"
|
|
local cursor_home_dir="$TARGET_HOME/.cursor"
|
|
|
|
# 确保目录存在
|
|
mkdir -p "$cursor_support_dir" 2>/dev/null || true
|
|
mkdir -p "$cursor_home_dir/extensions" 2>/dev/null || true
|
|
|
|
# 🔧 执行用户验证有效的4个核心权限修复命令
|
|
log_info "🔧 [修复] 执行4个核心权限修复命令..."
|
|
|
|
# 命令1: sudo chown -R <真实用户> ~/Library/"Application Support"/Cursor
|
|
if sudo chown -R "$target_user" "$cursor_support_dir" 2>/dev/null; then
|
|
log_info "✅ [1/4] sudo chown Application Support/Cursor 成功"
|
|
else
|
|
log_warn "⚠️ [1/4] sudo chown Application Support/Cursor 失败"
|
|
fi
|
|
|
|
# 命令2: sudo chown -R <真实用户> ~/.cursor
|
|
if sudo chown -R "$target_user" "$cursor_home_dir" 2>/dev/null; then
|
|
log_info "✅ [2/4] sudo chown .cursor 成功"
|
|
else
|
|
log_warn "⚠️ [2/4] sudo chown .cursor 失败"
|
|
fi
|
|
|
|
# 命令3: chmod -R u+rwX ~/Library/"Application Support"/Cursor
|
|
# - X:仅对目录(或原本有可执行位的文件)补 x,避免破坏文件权限
|
|
if chmod -R u+rwX "$cursor_support_dir" 2>/dev/null; then
|
|
log_info "✅ [3/4] chmod Application Support/Cursor 成功"
|
|
else
|
|
log_warn "⚠️ [3/4] chmod Application Support/Cursor 失败"
|
|
fi
|
|
|
|
# 命令4: chmod -R u+rwX ~/.cursor (修复整个目录,不仅仅是extensions子目录)
|
|
if chmod -R u+rwX "$cursor_home_dir" 2>/dev/null; then
|
|
log_info "✅ [4/4] chmod .cursor 成功"
|
|
else
|
|
log_warn "⚠️ [4/4] chmod .cursor 失败"
|
|
fi
|
|
|
|
log_info "✅ [完成] 核心权限修复命令执行完成"
|
|
return 0
|
|
}
|
|
|
|
# 关键权限修复函数(简化版本)
|
|
fix_cursor_permissions_critical() {
|
|
log_info "🚨 [关键权限修复] 执行权限修复..."
|
|
ensure_cursor_directory_permissions
|
|
}
|
|
|
|
# 🚀 Cursor启动前权限确保(简化版本)
|
|
ensure_cursor_startup_permissions() {
|
|
log_info "🚀 [启动前权限] 执行权限修复..."
|
|
ensure_cursor_directory_permissions
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
# 🛠️ 修改机器码配置(增强版)
|
|
modify_machine_code_config() {
|
|
local mode=${1:-"FULL"}
|
|
|
|
echo
|
|
log_info "🛠️ [配置] 正在修改机器码配置..."
|
|
|
|
local config_path="$TARGET_HOME/Library/Application Support/Cursor/User/globalStorage/storage.json"
|
|
|
|
# 增强的配置文件检查
|
|
if [ ! -f "$config_path" ]; then
|
|
log_error "❌ [错误] 配置文件不存在: $config_path"
|
|
echo
|
|
log_info "💡 [解决方案] 请尝试以下步骤:"
|
|
echo -e "${BLUE} 1️⃣ 手动启动Cursor应用程序${NC}"
|
|
echo -e "${BLUE} 2️⃣ 等待Cursor完全加载(约30秒)${NC}"
|
|
echo -e "${BLUE} 3️⃣ 关闭Cursor应用程序${NC}"
|
|
echo -e "${BLUE} 4️⃣ 重新运行此脚本${NC}"
|
|
echo
|
|
log_warn "⚠️ [备选方案] 如果问题持续:"
|
|
echo -e "${BLUE} • 选择脚本的'重置环境+修改机器码'选项${NC}"
|
|
echo -e "${BLUE} • 该选项会自动生成配置文件${NC}"
|
|
echo
|
|
|
|
# 提供用户选择
|
|
read -p "是否现在尝试启动Cursor生成配置文件?(y/n): " user_choice
|
|
if [[ "$user_choice" =~ ^(y|yes)$ ]]; then
|
|
log_info "🚀 [尝试] 正在尝试启动Cursor..."
|
|
if start_cursor_to_generate_config; then
|
|
return 0
|
|
fi
|
|
fi
|
|
|
|
return 1
|
|
fi
|
|
|
|
# 验证配置文件格式并显示结构
|
|
log_info "🔍 [验证] 检查配置文件格式..."
|
|
# 🔧 修复:避免把路径直接拼进 Python 源码字符串(路径包含引号等特殊字符时会导致语法错误)
|
|
if ! python3 -c 'import json, sys; json.load(open(sys.argv[1], "r", encoding="utf-8"))' "$config_path" 2>/dev/null; then
|
|
log_error "❌ [错误] 配置文件格式错误或损坏"
|
|
log_info "💡 [建议] 配置文件可能已损坏,建议选择'重置环境+修改机器码'选项"
|
|
return 1
|
|
fi
|
|
log_info "✅ [验证] 配置文件格式正确"
|
|
|
|
# 显示当前配置文件中的相关属性
|
|
log_info "📋 [当前配置] 检查现有的遥测属性:"
|
|
# 🔧 修复:避免把路径直接拼进 Python 源码字符串(路径包含引号等特殊字符时会导致语法错误)
|
|
python3 -c "
|
|
import json
|
|
import sys
|
|
try:
|
|
config_path = sys.argv[1]
|
|
with open(config_path, 'r', encoding='utf-8') as f:
|
|
config = json.load(f)
|
|
|
|
properties = ['telemetry.machineId', 'telemetry.macMachineId', 'telemetry.devDeviceId', 'telemetry.sqmId']
|
|
for prop in properties:
|
|
if prop in config:
|
|
value = config[prop]
|
|
display_value = value[:20] + '...' if len(value) > 20 else value
|
|
print(f' ✓ {prop} = {display_value}')
|
|
else:
|
|
print(f' - {prop} (不存在,将创建)')
|
|
except Exception as e:
|
|
print(f'Error reading config: {e}')
|
|
" "$config_path"
|
|
echo
|
|
|
|
# 显示操作进度
|
|
log_info "⏳ [进度] 1/5 - 生成新的设备标识符..."
|
|
|
|
# 生成新的ID
|
|
local MAC_MACHINE_ID=$(generate_uuid)
|
|
local UUID=$(generate_uuid)
|
|
local MACHINE_ID=$(generate_hex_bytes 32)
|
|
local SQM_ID="{$(generate_uuid | tr '[:lower:]' '[:upper:]')}"
|
|
# 🔧 新增: serviceMachineId (用于 storage.serviceMachineId)
|
|
local SERVICE_MACHINE_ID=$(generate_uuid)
|
|
# 🔧 新增: firstSessionDate (重置首次会话日期)
|
|
local FIRST_SESSION_DATE=$(date -u +"%Y-%m-%dT%H:%M:%S.000Z")
|
|
|
|
CURSOR_ID_MACHINE_ID="$MACHINE_ID"
|
|
CURSOR_ID_MAC_MACHINE_ID="$MAC_MACHINE_ID"
|
|
CURSOR_ID_DEVICE_ID="$UUID"
|
|
CURSOR_ID_SQM_ID="$SQM_ID"
|
|
CURSOR_ID_FIRST_SESSION_DATE="$FIRST_SESSION_DATE"
|
|
CURSOR_ID_SESSION_ID=$(generate_uuid)
|
|
CURSOR_ID_MAC_ADDRESS="${CURSOR_ID_MAC_ADDRESS:-00:11:22:33:44:55}"
|
|
|
|
log_info "✅ [进度] 1/5 - 设备标识符生成完成"
|
|
|
|
log_info "⏳ [进度] 2/5 - 创建备份目录..."
|
|
|
|
# 备份原始配置(增强版)
|
|
local backup_dir="$TARGET_HOME/Library/Application Support/Cursor/User/globalStorage/backups"
|
|
if ! mkdir -p "$backup_dir"; then
|
|
log_error "❌ [错误] 无法创建备份目录: $backup_dir"
|
|
return 1
|
|
fi
|
|
|
|
local backup_name="storage.json.backup_$(date +%Y%m%d_%H%M%S)"
|
|
local backup_path="$backup_dir/$backup_name"
|
|
|
|
log_info "⏳ [进度] 3/5 - 备份原始配置..."
|
|
if ! cp "$config_path" "$backup_path"; then
|
|
log_error "❌ [错误] 备份配置文件失败"
|
|
return 1
|
|
fi
|
|
|
|
# 验证备份是否成功
|
|
if [ -f "$backup_path" ]; then
|
|
local backup_size=$(wc -c < "$backup_path")
|
|
local original_size=$(wc -c < "$config_path")
|
|
if [ "$backup_size" -eq "$original_size" ]; then
|
|
log_info "✅ [进度] 3/5 - 配置备份成功: $backup_name"
|
|
else
|
|
log_warn "⚠️ [警告] 备份文件大小不匹配,但继续执行"
|
|
fi
|
|
else
|
|
log_error "❌ [错误] 备份文件创建失败"
|
|
return 1
|
|
fi
|
|
|
|
log_info "⏳ [进度] 4/5 - 更新配置文件..."
|
|
|
|
# 使用Python修改JSON配置(更可靠,安全方式)
|
|
# 🔧 修复:避免把路径/值直接拼进 Python 源码字符串(引号/反斜杠等特殊字符会导致语法错误或写入错误)
|
|
local python_result=$(python3 -c "
|
|
import json
|
|
import sys
|
|
|
|
try:
|
|
config_path = sys.argv[1]
|
|
machine_id = sys.argv[2]
|
|
mac_machine_id = sys.argv[3]
|
|
dev_device_id = sys.argv[4]
|
|
sqm_id = sys.argv[5]
|
|
service_machine_id = sys.argv[6]
|
|
first_session_date = sys.argv[7]
|
|
|
|
with open(config_path, 'r', encoding='utf-8') as f:
|
|
config = json.load(f)
|
|
|
|
# 安全更新配置,确保属性存在
|
|
# 🔧 修复: 添加 storage.serviceMachineId 和 telemetry.firstSessionDate
|
|
properties_to_update = {
|
|
'telemetry.machineId': machine_id,
|
|
'telemetry.macMachineId': mac_machine_id,
|
|
'telemetry.devDeviceId': dev_device_id,
|
|
'telemetry.sqmId': sqm_id,
|
|
'storage.serviceMachineId': service_machine_id,
|
|
'telemetry.firstSessionDate': first_session_date
|
|
}
|
|
|
|
for key, value in properties_to_update.items():
|
|
if key in config:
|
|
print(f' ✓ 更新属性: {key}')
|
|
else:
|
|
print(f' + 添加属性: {key}')
|
|
config[key] = value
|
|
|
|
with open(config_path, 'w', encoding='utf-8') as f:
|
|
json.dump(config, f, indent=2, ensure_ascii=False)
|
|
|
|
print('SUCCESS')
|
|
except Exception as e:
|
|
print(f'ERROR: {e}')
|
|
sys.exit(1)
|
|
" "$config_path" "$MACHINE_ID" "$MAC_MACHINE_ID" "$UUID" "$SQM_ID" "$SERVICE_MACHINE_ID" "$FIRST_SESSION_DATE" 2>&1)
|
|
|
|
# 🔧 关键修复:正确解析Python执行结果
|
|
local python_exit_code=$?
|
|
local python_success=false
|
|
|
|
# 检查Python脚本是否成功执行
|
|
if [ $python_exit_code -eq 0 ]; then
|
|
# 检查输出中是否包含SUCCESS标记(忽略其他输出)
|
|
if echo "$python_result" | grep -q "SUCCESS"; then
|
|
python_success=true
|
|
log_info "✅ [Python] 配置修改执行成功"
|
|
else
|
|
log_warn "⚠️ [Python] 执行成功但未找到SUCCESS标记"
|
|
log_info "💡 [调试] Python完整输出:"
|
|
echo "$python_result"
|
|
fi
|
|
else
|
|
log_error "❌ [Python] 脚本执行失败,退出码: $python_exit_code"
|
|
log_info "💡 [调试] Python完整输出:"
|
|
echo "$python_result"
|
|
fi
|
|
|
|
if [ "$python_success" = true ]; then
|
|
log_info "⏳ [进度] 5/5 - 验证修改结果..."
|
|
|
|
# 🔒 关键修复:在验证前确保文件权限正确
|
|
chmod 644 "$config_path" 2>/dev/null || true
|
|
|
|
# 验证修改是否成功
|
|
# 🔧 修复:避免把路径/值直接拼进 Python 源码字符串(引号/反斜杠等特殊字符会导致语法错误或写入错误)
|
|
local verification_result=$(python3 -c "
|
|
import json
|
|
import sys
|
|
try:
|
|
config_path = sys.argv[1]
|
|
machine_id = sys.argv[2]
|
|
mac_machine_id = sys.argv[3]
|
|
dev_device_id = sys.argv[4]
|
|
sqm_id = sys.argv[5]
|
|
service_machine_id = sys.argv[6]
|
|
first_session_date = sys.argv[7]
|
|
|
|
with open(config_path, 'r', encoding='utf-8') as f:
|
|
config = json.load(f)
|
|
|
|
# 🔧 修复: 添加 storage.serviceMachineId 和 telemetry.firstSessionDate 验证
|
|
properties_to_check = {
|
|
'telemetry.machineId': machine_id,
|
|
'telemetry.macMachineId': mac_machine_id,
|
|
'telemetry.devDeviceId': dev_device_id,
|
|
'telemetry.sqmId': sqm_id,
|
|
'storage.serviceMachineId': service_machine_id,
|
|
'telemetry.firstSessionDate': first_session_date
|
|
}
|
|
|
|
verification_passed = True
|
|
for key, expected_value in properties_to_check.items():
|
|
actual_value = config.get(key)
|
|
if actual_value == expected_value:
|
|
print(f'✓ {key}: 验证通过')
|
|
else:
|
|
print(f'✗ {key}: 验证失败 (期望: {expected_value}, 实际: {actual_value})')
|
|
verification_passed = False
|
|
|
|
if verification_passed:
|
|
print('VERIFICATION_SUCCESS')
|
|
else:
|
|
print('VERIFICATION_FAILED')
|
|
except Exception as e:
|
|
print(f'VERIFICATION_ERROR: {e}')
|
|
" "$config_path" "$MACHINE_ID" "$MAC_MACHINE_ID" "$UUID" "$SQM_ID" "$SERVICE_MACHINE_ID" "$FIRST_SESSION_DATE" 2>&1)
|
|
|
|
# 检查验证结果(忽略其他输出,只关注最终结果)
|
|
if echo "$verification_result" | grep -q "VERIFICATION_SUCCESS"; then
|
|
log_info "✅ [进度] 5/5 - 修改验证成功"
|
|
|
|
# 🔐 关键修复:设置配置文件为只读保护
|
|
if chmod 444 "$config_path" 2>/dev/null; then
|
|
log_info "🔐 [保护] 配置文件已设置为只读保护"
|
|
else
|
|
log_warn "⚠️ [警告] 无法设置配置文件只读保护"
|
|
fi
|
|
|
|
# 🛡️ 关键修复:执行权限修复
|
|
ensure_cursor_directory_permissions
|
|
|
|
echo
|
|
log_info "🎉 [成功] 机器码配置修改完成!"
|
|
log_info "📋 [详情] 已更新以下标识符:"
|
|
echo " 🔹 machineId: ${MACHINE_ID:0:20}..."
|
|
echo " 🔹 macMachineId: $MAC_MACHINE_ID"
|
|
echo " 🔹 devDeviceId: $UUID"
|
|
echo " 🔹 sqmId: $SQM_ID"
|
|
echo " 🔹 serviceMachineId: $SERVICE_MACHINE_ID"
|
|
echo " 🔹 firstSessionDate: $FIRST_SESSION_DATE"
|
|
echo
|
|
log_info "💾 [备份] 原配置已备份至: $backup_name"
|
|
|
|
# 🔧 新增: 修改 machineid 文件
|
|
log_info "🔧 [machineid] 正在修改 machineid 文件..."
|
|
local machineid_file_path="$TARGET_HOME/Library/Application Support/Cursor/machineid"
|
|
if [ -f "$machineid_file_path" ]; then
|
|
# 备份原始 machineid 文件
|
|
local machineid_backup="$backup_dir/machineid.backup_$(date +%Y%m%d_%H%M%S)"
|
|
cp "$machineid_file_path" "$machineid_backup" 2>/dev/null && \
|
|
log_info "💾 [备份] machineid 文件已备份: $machineid_backup"
|
|
fi
|
|
# 写入新的 serviceMachineId 到 machineid 文件
|
|
if echo -n "$SERVICE_MACHINE_ID" > "$machineid_file_path" 2>/dev/null; then
|
|
log_info "✅ [machineid] machineid 文件修改成功: $SERVICE_MACHINE_ID"
|
|
# 设置 machineid 文件为只读
|
|
chmod 444 "$machineid_file_path" 2>/dev/null && \
|
|
log_info "🔒 [保护] machineid 文件已设置为只读"
|
|
else
|
|
log_warn "⚠️ [machineid] machineid 文件修改失败"
|
|
log_info "💡 [提示] 可手动修改文件: $machineid_file_path"
|
|
fi
|
|
|
|
# 🔧 新增: 修改 .updaterId 文件(更新器设备标识符)
|
|
log_info "🔧 [updaterId] 正在修改 .updaterId 文件..."
|
|
local updater_id_file_path="$TARGET_HOME/Library/Application Support/Cursor/.updaterId"
|
|
if [ -f "$updater_id_file_path" ]; then
|
|
# 备份原始 .updaterId 文件
|
|
local updater_id_backup="$backup_dir/.updaterId.backup_$(date +%Y%m%d_%H%M%S)"
|
|
cp "$updater_id_file_path" "$updater_id_backup" 2>/dev/null && \
|
|
log_info "💾 [备份] .updaterId 文件已备份: $updater_id_backup"
|
|
fi
|
|
# 生成新的 updaterId(UUID格式)
|
|
local new_updater_id=$(generate_uuid)
|
|
if echo -n "$new_updater_id" > "$updater_id_file_path" 2>/dev/null; then
|
|
log_info "✅ [updaterId] .updaterId 文件修改成功: $new_updater_id"
|
|
# 设置 .updaterId 文件为只读
|
|
chmod 444 "$updater_id_file_path" 2>/dev/null && \
|
|
log_info "🔒 [保护] .updaterId 文件已设置为只读"
|
|
else
|
|
log_warn "⚠️ [updaterId] .updaterId 文件修改失败"
|
|
log_info "💡 [提示] 可手动修改文件: $updater_id_file_path"
|
|
fi
|
|
|
|
return 0
|
|
else
|
|
log_error "❌ [错误] 修改验证失败"
|
|
log_info "💡 [验证详情]:"
|
|
echo "$verification_result"
|
|
log_info "🔄 [恢复] 正在恢复备份并修复权限..."
|
|
|
|
# 恢复备份并确保权限正确
|
|
if cp "$backup_path" "$config_path"; then
|
|
chmod 644 "$config_path" 2>/dev/null || true
|
|
ensure_cursor_directory_permissions
|
|
log_info "✅ [恢复] 已恢复原始配置并修复权限"
|
|
else
|
|
log_error "❌ [错误] 恢复备份失败"
|
|
fi
|
|
return 1
|
|
fi
|
|
else
|
|
log_error "❌ [错误] 修改配置失败"
|
|
log_info "💡 [调试信息] Python执行详情:"
|
|
echo "$python_result"
|
|
|
|
# 尝试恢复备份并修复权限
|
|
if [ -f "$backup_path" ]; then
|
|
log_info "🔄 [恢复] 正在恢复备份配置并修复权限..."
|
|
if cp "$backup_path" "$config_path"; then
|
|
chmod 644 "$config_path" 2>/dev/null || true
|
|
ensure_cursor_directory_permissions
|
|
log_info "✅ [恢复] 已恢复原始配置并修复权限"
|
|
else
|
|
log_error "❌ [错误] 恢复备份失败"
|
|
fi
|
|
fi
|
|
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
|
|
|
|
# 获取当前用户
|
|
get_current_user() {
|
|
if [ "$EUID" -eq 0 ]; then
|
|
echo "$SUDO_USER"
|
|
else
|
|
echo "$USER"
|
|
fi
|
|
}
|
|
|
|
# 获取指定用户的 Home 目录(用于 sudo 环境下仍能定位到真实用户目录)
|
|
get_user_home_dir() {
|
|
local user="$1"
|
|
local home_dir=""
|
|
|
|
if [ -z "$user" ]; then
|
|
echo ""
|
|
return 1
|
|
fi
|
|
|
|
# macOS:优先使用 dscl,避免 sudo -H / env_reset 影响 $HOME
|
|
if command -v dscl >/dev/null 2>&1; then
|
|
home_dir=$(dscl . -read "/Users/$user" NFSHomeDirectory 2>/dev/null | awk '{print $2}')
|
|
fi
|
|
|
|
# 回退:使用 shell 的 ~ 展开(某些环境 dscl 读取可能失败)
|
|
if [ -z "$home_dir" ]; then
|
|
home_dir=$(eval echo "~$user" 2>/dev/null)
|
|
fi
|
|
|
|
# 最终回退:当前环境的 $HOME(至少保证脚本不因空值崩溃)
|
|
if [ -z "$home_dir" ]; then
|
|
home_dir="$HOME"
|
|
fi
|
|
|
|
echo "$home_dir"
|
|
return 0
|
|
}
|
|
|
|
CURRENT_USER=$(get_current_user)
|
|
if [ -z "$CURRENT_USER" ]; then
|
|
log_error "无法获取用户名"
|
|
exit 1
|
|
fi
|
|
|
|
# 🎯 统一“目标用户/目标 Home”:后续所有 Cursor 用户数据路径均基于该 Home
|
|
TARGET_USER="$CURRENT_USER"
|
|
TARGET_HOME="$(get_user_home_dir "$TARGET_USER")"
|
|
|
|
# 定义配置文件路径
|
|
STORAGE_FILE="$TARGET_HOME/Library/Application Support/Cursor/User/globalStorage/storage.json"
|
|
BACKUP_DIR="$TARGET_HOME/Library/Application Support/Cursor/User/globalStorage/backups"
|
|
|
|
# 共享ID(用于配置与JS注入保持一致)
|
|
CURSOR_ID_MACHINE_ID=""
|
|
CURSOR_ID_MACHINE_GUID=""
|
|
CURSOR_ID_MAC_MACHINE_ID=""
|
|
CURSOR_ID_DEVICE_ID=""
|
|
CURSOR_ID_SQM_ID=""
|
|
CURSOR_ID_FIRST_SESSION_DATE=""
|
|
CURSOR_ID_SESSION_ID=""
|
|
CURSOR_ID_MAC_ADDRESS="00:11:22:33:44:55"
|
|
|
|
# 定义 Cursor 应用程序路径
|
|
CURSOR_APP_PATH="/Applications/Cursor.app"
|
|
|
|
# 新增:判断接口类型是否为Wi-Fi
|
|
is_wifi_interface() {
|
|
local interface_name="$1"
|
|
# 通过networksetup判断接口类型
|
|
networksetup -listallhardwareports | \
|
|
awk -v dev="$interface_name" 'BEGIN{found=0} /Hardware Port: Wi-Fi/{found=1} /Device:/{if(found && $2==dev){exit 0}else{found=0}}' && return 0 || return 1
|
|
}
|
|
|
|
# 🎯 增强的MAC地址生成和验证(集成randommac.sh特性)
|
|
generate_local_unicast_mac() {
|
|
# 第一字节:LAA+单播(低两位10),其余随机
|
|
local first_byte=$(( (RANDOM & 0xFC) | 0x02 ))
|
|
local mac=$(printf '%02x:%02x:%02x:%02x:%02x:%02x' \
|
|
$first_byte $((RANDOM%256)) $((RANDOM%256)) $((RANDOM%256)) $((RANDOM%256)) $((RANDOM%256)))
|
|
echo "$mac"
|
|
}
|
|
|
|
# 🔍 MAC地址验证函数(基于randommac.sh)
|
|
validate_mac_address() {
|
|
local mac="$1"
|
|
local regex="^([0-9A-Fa-f]{2}[:]){5}([0-9A-Fa-f]{2})$"
|
|
|
|
if [[ $mac =~ $regex ]]; then
|
|
return 0
|
|
else
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
|
|
|
|
# 🔄 增强的WiFi断开和重连机制
|
|
manage_wifi_connection() {
|
|
local action="$1" # disconnect 或 reconnect
|
|
local interface_name="$2"
|
|
|
|
if ! is_wifi_interface "$interface_name"; then
|
|
log_info "📡 [跳过] 接口 '$interface_name' 不是WiFi,跳过WiFi管理"
|
|
return 0
|
|
fi
|
|
|
|
case "$action" in
|
|
"disconnect")
|
|
log_info "📡 [WiFi] 断开WiFi连接但保持适配器开启..."
|
|
|
|
# 方法1: 使用airport工具断开
|
|
if command -v /System/Library/PrivateFrameworks/Apple80211.framework/Versions/Current/Resources/airport >/dev/null 2>&1; then
|
|
sudo /System/Library/PrivateFrameworks/Apple80211.framework/Versions/Current/Resources/airport -z 2>>"$LOG_FILE"
|
|
log_info "✅ [WiFi] 使用airport工具断开WiFi连接"
|
|
else
|
|
# 方法2: 使用networksetup断开
|
|
local wifi_service=$(networksetup -listallhardwareports | grep -A1 "Device: $interface_name" | grep "Hardware Port:" | cut -d: -f2 | xargs)
|
|
if [ -n "$wifi_service" ]; then
|
|
networksetup -setairportpower "$interface_name" off 2>>"$LOG_FILE"
|
|
sleep 2
|
|
networksetup -setairportpower "$interface_name" on 2>>"$LOG_FILE"
|
|
log_info "✅ [WiFi] 使用networksetup重置WiFi适配器"
|
|
else
|
|
log_warn "⚠️ [WiFi] 无法找到WiFi服务,跳过断开"
|
|
fi
|
|
fi
|
|
|
|
sleep 3
|
|
;;
|
|
|
|
"reconnect")
|
|
log_info "📡 [WiFi] 重新连接WiFi..."
|
|
|
|
# 触发网络硬件重新检测
|
|
sudo networksetup -detectnewhardware 2>>"$LOG_FILE"
|
|
|
|
# 等待网络重新连接
|
|
log_info "⏳ [WiFi] 等待WiFi重新连接..."
|
|
local wait_count=0
|
|
local max_wait=30
|
|
|
|
while [ $wait_count -lt $max_wait ]; do
|
|
if ping -c 1 8.8.8.8 >/dev/null 2>&1; then
|
|
log_info "✅ [WiFi] 网络连接已恢复"
|
|
return 0
|
|
fi
|
|
sleep 2
|
|
wait_count=$((wait_count + 2))
|
|
|
|
if [ $((wait_count % 10)) -eq 0 ]; then
|
|
log_info "⏳ [WiFi] 等待网络连接... ($wait_count/$max_wait 秒)"
|
|
fi
|
|
done
|
|
|
|
log_warn "⚠️ [WiFi] 网络连接未在预期时间内恢复,但继续执行"
|
|
;;
|
|
|
|
*)
|
|
log_error "❌ [错误] 无效的WiFi管理操作: $action"
|
|
return 1
|
|
;;
|
|
esac
|
|
}
|
|
|
|
# 🛠️ 增强的第三方工具MAC地址修改
|
|
try_third_party_mac_tool() {
|
|
local interface_name="$1"
|
|
local random_mac="$2"
|
|
local success=false
|
|
local tool_used=""
|
|
|
|
log_info "🛠️ [第三方] 尝试使用第三方工具修改MAC地址"
|
|
|
|
# 🔍 检测可用的第三方工具
|
|
local available_tools=()
|
|
if command -v macchanger >/dev/null 2>&1; then
|
|
available_tools+=("macchanger")
|
|
fi
|
|
if command -v spoof-mac >/dev/null 2>&1; then
|
|
available_tools+=("spoof-mac")
|
|
fi
|
|
|
|
if [ ${#available_tools[@]} -eq 0 ]; then
|
|
log_warn "⚠️ [警告] 未检测到可用的第三方MAC地址修改工具"
|
|
log_info "💡 [建议] 可以安装以下工具:"
|
|
echo " • brew install spoof-mac"
|
|
echo " • brew install macchanger"
|
|
return 1
|
|
fi
|
|
|
|
log_info "🔍 [检测] 发现可用工具: ${available_tools[*]}"
|
|
|
|
# 🎯 优先使用macchanger
|
|
if [[ " ${available_tools[*]} " =~ " macchanger " ]]; then
|
|
log_info "🔧 [macchanger] 尝试使用macchanger修改接口 '$interface_name' 的MAC地址..."
|
|
|
|
# 先关闭接口
|
|
sudo ifconfig "$interface_name" down 2>>"$LOG_FILE"
|
|
sleep 2
|
|
|
|
if sudo macchanger -m "$random_mac" "$interface_name" >>"$LOG_FILE" 2>&1; then
|
|
success=true
|
|
tool_used="macchanger"
|
|
log_info "✅ [成功] macchanger修改成功"
|
|
else
|
|
log_warn "⚠️ [失败] macchanger修改失败"
|
|
fi
|
|
|
|
# 重新启用接口
|
|
sudo ifconfig "$interface_name" up 2>>"$LOG_FILE"
|
|
sleep 2
|
|
fi
|
|
|
|
# 🎯 如果macchanger失败,尝试spoof-mac
|
|
if ! $success && [[ " ${available_tools[*]} " =~ " spoof-mac " ]]; then
|
|
log_info "🔧 [spoof-mac] 尝试使用spoof-mac修改接口 '$interface_name' 的MAC地址..."
|
|
|
|
if sudo spoof-mac set "$random_mac" "$interface_name" >>"$LOG_FILE" 2>&1; then
|
|
success=true
|
|
tool_used="spoof-mac"
|
|
log_info "✅ [成功] spoof-mac修改成功"
|
|
else
|
|
log_warn "⚠️ [失败] spoof-mac修改失败"
|
|
fi
|
|
fi
|
|
|
|
if $success; then
|
|
log_info "🎉 [成功] 第三方工具 ($tool_used) 修改MAC地址成功"
|
|
return 0
|
|
else
|
|
log_error "❌ [失败] 所有第三方工具都修改失败"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 🔍 增强的macOS环境检测和兼容性评估
|
|
detect_macos_environment() {
|
|
local macos_version=$(sw_vers -productVersion)
|
|
local macos_major=$(echo "$macos_version" | cut -d. -f1)
|
|
local macos_minor=$(echo "$macos_version" | cut -d. -f2)
|
|
local hardware_type=""
|
|
|
|
# 检测硬件类型
|
|
if [[ $(uname -m) == "arm64" ]]; then
|
|
hardware_type="Apple Silicon"
|
|
else
|
|
hardware_type="Intel"
|
|
fi
|
|
|
|
log_info "🔍 [环境] 系统环境检测: macOS $macos_version ($hardware_type)"
|
|
|
|
# 检查SIP状态
|
|
local sip_status=$(csrutil status 2>/dev/null | grep -o "enabled\|disabled" || echo "unknown")
|
|
log_info "🔒 [SIP] 系统完整性保护状态: $sip_status"
|
|
|
|
# 设置环境变量
|
|
export MACOS_VERSION="$macos_version"
|
|
export MACOS_MAJOR="$macos_major"
|
|
export MACOS_MINOR="$macos_minor"
|
|
export HARDWARE_TYPE="$hardware_type"
|
|
export SIP_STATUS="$sip_status"
|
|
|
|
# 🎯 增强的兼容性检查
|
|
local compatibility_level="FULL"
|
|
local compatibility_issues=()
|
|
|
|
# 检查macOS版本兼容性
|
|
if [[ $macos_major -ge 14 ]]; then
|
|
compatibility_issues+=("macOS $macos_major+ 对MAC地址修改有严格限制")
|
|
compatibility_level="LIMITED"
|
|
elif [[ $macos_major -ge 12 ]]; then
|
|
compatibility_issues+=("macOS $macos_major 可能对MAC地址修改有部分限制")
|
|
compatibility_level="PARTIAL"
|
|
fi
|
|
|
|
# 检查硬件兼容性
|
|
if [[ "$hardware_type" == "Apple Silicon" ]]; then
|
|
compatibility_issues+=("Apple Silicon硬件对MAC地址修改有硬件级限制")
|
|
if [[ "$compatibility_level" == "FULL" ]]; then
|
|
compatibility_level="PARTIAL"
|
|
else
|
|
compatibility_level="MINIMAL"
|
|
fi
|
|
fi
|
|
|
|
# 检查SIP影响
|
|
if [[ "$sip_status" == "enabled" ]]; then
|
|
compatibility_issues+=("系统完整性保护(SIP)可能阻止某些修改方法")
|
|
fi
|
|
|
|
# 设置兼容性级别
|
|
export MAC_COMPATIBILITY_LEVEL="$compatibility_level"
|
|
|
|
# 显示兼容性评估结果
|
|
case "$compatibility_level" in
|
|
"FULL")
|
|
log_info "✅ [兼容性] 完全兼容 - 支持所有MAC地址修改方法"
|
|
;;
|
|
"PARTIAL")
|
|
log_warn "⚠️ [兼容性] 部分兼容 - 某些方法可能失败"
|
|
;;
|
|
"LIMITED")
|
|
log_warn "⚠️ [兼容性] 有限兼容 - 大多数方法可能失败"
|
|
;;
|
|
"MINIMAL")
|
|
log_error "❌ [兼容性] 最小兼容 - MAC地址修改可能完全失败"
|
|
;;
|
|
esac
|
|
|
|
if [ ${#compatibility_issues[@]} -gt 0 ]; then
|
|
log_info "📋 [兼容性问题]:"
|
|
for issue in "${compatibility_issues[@]}"; do
|
|
echo " • $issue"
|
|
done
|
|
fi
|
|
|
|
# 返回兼容性状态
|
|
case "$compatibility_level" in
|
|
"FULL"|"PARTIAL") return 0 ;;
|
|
*) return 1 ;;
|
|
esac
|
|
}
|
|
|
|
# 🚀 增强的MAC地址修改函数,支持智能方法选择
|
|
_change_mac_for_one_interface() {
|
|
local interface_name="$1"
|
|
|
|
if [ -z "$interface_name" ]; then
|
|
log_error "❌ [错误] _change_mac_for_one_interface: 未提供接口名称"
|
|
return 1
|
|
fi
|
|
|
|
log_info "🚀 [开始] 开始处理接口: $interface_name"
|
|
echo
|
|
|
|
# 🔍 环境检测和兼容性评估
|
|
detect_macos_environment
|
|
local env_compatible=$?
|
|
local compatibility_level="$MAC_COMPATIBILITY_LEVEL"
|
|
|
|
# 📡 获取当前MAC地址
|
|
local current_mac=$(ifconfig "$interface_name" | awk '/ether/{print $2}')
|
|
if [ -z "$current_mac" ]; then
|
|
log_warn "⚠️ [警告] 无法获取接口 '$interface_name' 的当前MAC地址,可能已禁用或不存在"
|
|
return 1
|
|
else
|
|
log_info "📍 [当前] 接口 '$interface_name' 当前MAC地址: $current_mac"
|
|
fi
|
|
|
|
# 🎯 自动生成新MAC地址
|
|
local random_mac=$(generate_local_unicast_mac)
|
|
log_info "🎲 [生成] 为接口 '$interface_name' 生成新MAC地址: $random_mac"
|
|
|
|
# 📋 显示修改计划
|
|
echo
|
|
log_info "📋 [计划] MAC地址修改计划:"
|
|
echo " 🔹 接口: $interface_name"
|
|
echo " 🔹 当前MAC: $current_mac"
|
|
echo " 🔹 目标MAC: $random_mac"
|
|
echo " 🔹 兼容性: $compatibility_level"
|
|
echo
|
|
|
|
# 🔄 WiFi预处理
|
|
manage_wifi_connection "disconnect" "$interface_name"
|
|
|
|
# 🛠️ 执行MAC地址修改(多方法尝试)
|
|
local mac_change_success=false
|
|
local method_used=""
|
|
local methods_tried=()
|
|
|
|
# 📊 根据兼容性级别选择方法顺序
|
|
local method_order=()
|
|
case "$compatibility_level" in
|
|
"FULL")
|
|
method_order=("ifconfig" "third-party" "networksetup")
|
|
;;
|
|
"PARTIAL")
|
|
method_order=("third-party" "ifconfig" "networksetup")
|
|
;;
|
|
"LIMITED"|"MINIMAL")
|
|
method_order=("third-party" "networksetup" "ifconfig")
|
|
;;
|
|
esac
|
|
|
|
log_info "🛠️ [方法] 将按以下顺序尝试修改方法: ${method_order[*]}"
|
|
echo
|
|
|
|
# 🔄 逐个尝试修改方法
|
|
for method in "${method_order[@]}"; do
|
|
log_info "🔧 [尝试] 正在尝试 $method 方法..."
|
|
methods_tried+=("$method")
|
|
|
|
case "$method" in
|
|
"ifconfig")
|
|
if _try_ifconfig_method "$interface_name" "$random_mac"; then
|
|
mac_change_success=true
|
|
method_used="ifconfig"
|
|
break
|
|
fi
|
|
;;
|
|
"third-party")
|
|
if try_third_party_mac_tool "$interface_name" "$random_mac"; then
|
|
mac_change_success=true
|
|
method_used="third-party"
|
|
break
|
|
fi
|
|
;;
|
|
"networksetup")
|
|
if _try_networksetup_method "$interface_name" "$random_mac"; then
|
|
mac_change_success=true
|
|
method_used="networksetup"
|
|
break
|
|
fi
|
|
;;
|
|
esac
|
|
|
|
log_warn "⚠️ [失败] $method 方法失败,尝试下一个方法..."
|
|
sleep 2
|
|
done
|
|
|
|
# 🔍 验证修改结果
|
|
if [[ $mac_change_success == true ]]; then
|
|
log_info "🔍 [验证] 验证MAC地址修改结果..."
|
|
sleep 3 # 等待系统更新
|
|
|
|
local final_mac_check=$(ifconfig "$interface_name" | awk '/ether/{print $2}')
|
|
log_info "📍 [检查] 接口 '$interface_name' 最终MAC地址: $final_mac_check"
|
|
|
|
if [ "$final_mac_check" == "$random_mac" ]; then
|
|
echo
|
|
log_info "🎉 [成功] MAC地址修改成功!"
|
|
echo " ✅ 使用方法: $method_used"
|
|
echo " ✅ 接口: $interface_name"
|
|
echo " ✅ 原MAC: $current_mac"
|
|
echo " ✅ 新MAC: $final_mac_check"
|
|
|
|
# 🔄 WiFi后处理
|
|
manage_wifi_connection "reconnect" "$interface_name"
|
|
|
|
return 0
|
|
else
|
|
log_warn "⚠️ [验证失败] MAC地址可能未生效或已被系统重置"
|
|
log_info "💡 [提示] 期望: $random_mac, 实际: $final_mac_check"
|
|
mac_change_success=false
|
|
fi
|
|
fi
|
|
|
|
# ❌ 失败处理和用户选择
|
|
if [[ $mac_change_success == false ]]; then
|
|
echo
|
|
log_error "❌ [失败] 所有MAC地址修改方法都失败了"
|
|
log_info "📋 [尝试过的方法]: ${methods_tried[*]}"
|
|
|
|
# 🔄 WiFi恢复
|
|
manage_wifi_connection "reconnect" "$interface_name"
|
|
|
|
# 📊 显示故障排除信息
|
|
_show_troubleshooting_info "$interface_name"
|
|
|
|
# 🎯 提供用户选择
|
|
echo
|
|
echo -e "${BLUE}💡 [说明]${NC} MAC地址修改失败,您可以选择:"
|
|
echo -e "${BLUE}💡 [备注]${NC} 如果所有接口都失败,脚本会自动尝试JS内核修改方案"
|
|
echo
|
|
|
|
# 简化的用户选择
|
|
echo "请选择操作:"
|
|
echo " 1. 重试本接口"
|
|
echo " 2. 跳过本接口"
|
|
echo " 3. 退出脚本"
|
|
|
|
read -p "请输入选择 (1-3): " choice
|
|
|
|
case "$choice" in
|
|
1)
|
|
log_info "🔄 [重试] 用户选择重试本接口"
|
|
_change_mac_for_one_interface "$interface_name"
|
|
;;
|
|
2)
|
|
log_info "⏭️ [跳过] 用户选择跳过本接口"
|
|
return 1
|
|
;;
|
|
3)
|
|
log_info "🚪 [退出] 用户选择退出脚本"
|
|
exit 1
|
|
;;
|
|
*)
|
|
log_info "⏭️ [默认] 无效选择,跳过本接口"
|
|
return 1
|
|
;;
|
|
esac
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 🔧 增强的传统ifconfig方法(集成WiFi管理)
|
|
_try_ifconfig_method() {
|
|
local interface_name="$1"
|
|
local random_mac="$2"
|
|
|
|
log_info "🔧 [ifconfig] 使用传统ifconfig方法修改MAC地址"
|
|
|
|
# 🔄 WiFi特殊处理已在主函数中处理,这里只需要基本的接口操作
|
|
log_info "📡 [接口] 临时禁用接口 '$interface_name' 以修改MAC地址..."
|
|
if ! sudo ifconfig "$interface_name" down 2>>"$LOG_FILE"; then
|
|
log_error "❌ [错误] 禁用接口 '$interface_name' 失败"
|
|
return 1
|
|
fi
|
|
|
|
log_info "⏳ [等待] 等待接口完全关闭..."
|
|
sleep 3
|
|
|
|
# 🎯 尝试修改MAC地址
|
|
log_info "🎯 [修改] 设置新MAC地址: $random_mac"
|
|
if sudo ifconfig "$interface_name" ether "$random_mac" 2>>"$LOG_FILE"; then
|
|
log_info "✅ [成功] MAC地址设置命令执行成功"
|
|
|
|
# 重新启用接口
|
|
log_info "🔄 [启用] 重新启用接口..."
|
|
if sudo ifconfig "$interface_name" up 2>>"$LOG_FILE"; then
|
|
log_info "✅ [成功] 接口重新启用成功"
|
|
sleep 2
|
|
return 0
|
|
else
|
|
log_error "❌ [错误] 重新启用接口失败"
|
|
return 1
|
|
fi
|
|
else
|
|
log_error "❌ [错误] ifconfig ether 命令失败"
|
|
log_info "🔄 [恢复] 尝试重新启用接口..."
|
|
sudo ifconfig "$interface_name" up 2>/dev/null || true
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 🌐 增强的networksetup方法(适用于受限环境)
|
|
_try_networksetup_method() {
|
|
local interface_name="$1"
|
|
local random_mac="$2"
|
|
|
|
log_info "🌐 [networksetup] 尝试使用系统网络偏好设置方法"
|
|
|
|
# 🔍 获取硬件端口名称
|
|
local hardware_port=$(networksetup -listallhardwareports | grep -A1 "Device: $interface_name" | grep "Hardware Port:" | cut -d: -f2 | xargs)
|
|
|
|
if [ -z "$hardware_port" ]; then
|
|
log_warn "⚠️ [警告] 无法找到接口 $interface_name 对应的硬件端口"
|
|
log_info "📋 [调试] 可用硬件端口列表:"
|
|
networksetup -listallhardwareports | grep -E "(Hardware Port|Device)" | head -10
|
|
return 1
|
|
fi
|
|
|
|
log_info "🔍 [发现] 找到硬件端口: '$hardware_port' (设备: $interface_name)"
|
|
|
|
# 🎯 尝试多种networksetup方法
|
|
local methods_tried=()
|
|
|
|
# 方法1: 尝试重置网络服务
|
|
log_info "🔧 [方法1] 尝试重置网络服务..."
|
|
methods_tried+=("reset-service")
|
|
if sudo networksetup -setnetworkserviceenabled "$hardware_port" off 2>>"$LOG_FILE"; then
|
|
sleep 2
|
|
if sudo networksetup -setnetworkserviceenabled "$hardware_port" on 2>>"$LOG_FILE"; then
|
|
log_info "✅ [成功] 网络服务重置成功"
|
|
sleep 2
|
|
|
|
# 检测硬件变化
|
|
sudo networksetup -detectnewhardware 2>>"$LOG_FILE"
|
|
sleep 3
|
|
|
|
# 验证是否有效果
|
|
local new_mac=$(ifconfig "$interface_name" | awk '/ether/{print $2}')
|
|
if [ "$new_mac" != "$(ifconfig "$interface_name" | awk '/ether/{print $2}')" ]; then
|
|
log_info "✅ [成功] networksetup方法可能有效"
|
|
return 0
|
|
fi
|
|
fi
|
|
fi
|
|
|
|
# 方法2: 尝试手动配置
|
|
log_info "🔧 [方法2] 尝试手动网络配置..."
|
|
methods_tried+=("manual-config")
|
|
|
|
# 获取当前配置
|
|
local current_config=$(networksetup -getinfo "$hardware_port" 2>/dev/null)
|
|
if [ -n "$current_config" ]; then
|
|
log_info "📋 [当前配置] $hardware_port 的网络配置:"
|
|
echo "$current_config" | head -5
|
|
|
|
# 尝试重新应用配置以触发MAC地址更新
|
|
if echo "$current_config" | grep -q "DHCP"; then
|
|
log_info "🔄 [DHCP] 重新应用DHCP配置..."
|
|
if sudo networksetup -setdhcp "$hardware_port" 2>>"$LOG_FILE"; then
|
|
log_info "✅ [成功] DHCP配置重新应用成功"
|
|
sleep 3
|
|
sudo networksetup -detectnewhardware 2>>"$LOG_FILE"
|
|
return 0
|
|
fi
|
|
fi
|
|
fi
|
|
|
|
# 方法3: 强制硬件重新检测
|
|
log_info "🔧 [方法3] 强制硬件重新检测..."
|
|
methods_tried+=("hardware-detect")
|
|
|
|
if sudo networksetup -detectnewhardware 2>>"$LOG_FILE"; then
|
|
log_info "✅ [成功] 硬件重新检测完成"
|
|
sleep 3
|
|
return 0
|
|
fi
|
|
|
|
# 所有方法都失败
|
|
log_error "❌ [失败] networksetup所有方法都失败"
|
|
log_info "📋 [尝试过的方法]: ${methods_tried[*]}"
|
|
log_warn "⚠️ [说明] networksetup方法在当前macOS版本中可能不支持直接MAC地址修改"
|
|
|
|
return 1
|
|
}
|
|
|
|
# 📊 增强的故障排除信息显示
|
|
_show_troubleshooting_info() {
|
|
local interface_name="$1"
|
|
|
|
echo
|
|
echo -e "${YELLOW}╔══════════════════════════════════════════════════════════════╗${NC}"
|
|
echo -e "${YELLOW}║ MAC地址修改故障排除信息 ║${NC}"
|
|
echo -e "${YELLOW}╚══════════════════════════════════════════════════════════════╝${NC}"
|
|
echo
|
|
|
|
# 🔍 系统信息
|
|
echo -e "${BLUE}🔍 系统环境信息:${NC}"
|
|
echo " 📱 macOS版本: $MACOS_VERSION"
|
|
echo " 💻 硬件类型: $HARDWARE_TYPE"
|
|
echo " 🔒 SIP状态: $SIP_STATUS"
|
|
echo " 🌐 接口名称: $interface_name"
|
|
echo " 📊 兼容性级别: ${MAC_COMPATIBILITY_LEVEL:-未知}"
|
|
|
|
# 显示接口详细信息
|
|
local interface_info=$(ifconfig "$interface_name" 2>/dev/null | head -3)
|
|
if [ -n "$interface_info" ]; then
|
|
echo " 📡 接口状态:"
|
|
echo "$interface_info" | sed 's/^/ /'
|
|
fi
|
|
echo
|
|
|
|
# ⚠️ 问题分析
|
|
echo -e "${BLUE}⚠️ 可能的问题原因:${NC}"
|
|
local issues_found=false
|
|
|
|
if [[ "$HARDWARE_TYPE" == "Apple Silicon" ]] && [[ $MACOS_MAJOR -ge 12 ]]; then
|
|
echo " ❌ Apple Silicon Mac在macOS 12+版本中有硬件级MAC地址修改限制"
|
|
echo " ❌ 网络驱动程序可能完全禁止MAC地址修改"
|
|
issues_found=true
|
|
fi
|
|
|
|
if [[ $MACOS_MAJOR -ge 14 ]]; then
|
|
echo " ❌ macOS Sonoma (14+) 对MAC地址修改有严格的系统级限制"
|
|
issues_found=true
|
|
elif [[ $MACOS_MAJOR -ge 12 ]]; then
|
|
echo " ⚠️ macOS Monterey+ 对MAC地址修改有部分限制"
|
|
issues_found=true
|
|
fi
|
|
|
|
if [[ "$SIP_STATUS" == "enabled" ]]; then
|
|
echo " ⚠️ 系统完整性保护(SIP)可能阻止某些MAC地址修改方法"
|
|
issues_found=true
|
|
fi
|
|
|
|
if ! $issues_found; then
|
|
echo " ❓ 网络接口可能不支持MAC地址修改"
|
|
echo " ❓ 权限不足或其他系统安全策略限制"
|
|
fi
|
|
echo
|
|
|
|
# 💡 解决方案
|
|
echo -e "${BLUE}💡 建议的解决方案:${NC}"
|
|
echo
|
|
echo -e "${GREEN} 🛠️ 方案1: 安装第三方工具${NC}"
|
|
echo " brew install spoof-mac"
|
|
echo " brew install macchanger"
|
|
echo " # 这些工具可能使用不同的底层方法"
|
|
echo
|
|
|
|
if [[ "$HARDWARE_TYPE" == "Apple Silicon" ]] || [[ $MACOS_MAJOR -ge 14 ]]; then
|
|
echo -e "${GREEN} 🔧 方案2: 使用Cursor JS内核修改 (推荐)${NC}"
|
|
echo " # 本脚本会自动尝试JS内核修改方案"
|
|
echo " # 直接修改Cursor内核文件绕过系统MAC检测"
|
|
echo
|
|
fi
|
|
|
|
echo -e "${GREEN} 🌐 方案3: 网络层解决方案${NC}"
|
|
echo " • 使用虚拟机运行需要MAC地址修改的应用"
|
|
echo " • 配置路由器级别的MAC地址过滤绕过"
|
|
echo " • 使用VPN或代理服务"
|
|
echo
|
|
|
|
if [[ "$SIP_STATUS" == "enabled" ]]; then
|
|
echo -e "${YELLOW} ⚠️ 方案4: 临时禁用SIP (高风险,不推荐)${NC}"
|
|
echo " 1. 重启进入恢复模式 (Command+R)"
|
|
echo " 2. 打开终端运行: csrutil disable"
|
|
echo " 3. 重启后尝试修改MAC地址"
|
|
echo " 4. 完成后重新启用: csrutil enable"
|
|
echo " ⚠️ 警告: 禁用SIP会降低系统安全性"
|
|
echo
|
|
fi
|
|
|
|
# 🔧 技术细节
|
|
echo -e "${BLUE}🔧 技术细节和错误分析:${NC}"
|
|
echo " 📋 常见错误信息:"
|
|
echo " • ifconfig: ioctl (SIOCAIFADDR): Can't assign requested address"
|
|
echo " • Operation not permitted"
|
|
echo " • Device or resource busy"
|
|
echo
|
|
echo " 🔍 错误含义:"
|
|
echo " • 系统内核拒绝了MAC地址修改请求"
|
|
echo " • 硬件驱动程序不允许MAC地址更改"
|
|
echo " • 安全策略阻止了网络接口修改"
|
|
echo
|
|
|
|
if [[ "$HARDWARE_TYPE" == "Apple Silicon" ]]; then
|
|
echo " 🍎 Apple Silicon特殊说明:"
|
|
echo " • 硬件级别的安全限制,无法通过软件绕过"
|
|
echo " • 网络芯片固件可能锁定了MAC地址"
|
|
echo " • 建议使用应用层解决方案(如JS内核修改)"
|
|
echo
|
|
fi
|
|
|
|
echo -e "${BLUE}📞 获取更多帮助:${NC}"
|
|
echo " • 查看系统日志: sudo dmesg | grep -i network"
|
|
echo " • 检查网络接口: networksetup -listallhardwareports"
|
|
echo " • 测试权限: sudo ifconfig $interface_name"
|
|
echo
|
|
}
|
|
|
|
# 智能设备识别绕过(已移除 MAC 地址修改,仅保留 JS 注入)
|
|
run_device_bypass() {
|
|
log_info "🔧 [设备识别] 已禁用 MAC 地址修改,直接执行 JS 内核注入..."
|
|
|
|
if modify_cursor_js_files; then
|
|
log_info "✅ [JS] JS 内核注入完成"
|
|
return 0
|
|
fi
|
|
|
|
log_error "❌ [JS] JS 内核注入失败"
|
|
return 1
|
|
}
|
|
|
|
# 检查权限
|
|
check_permissions() {
|
|
if [ "$EUID" -ne 0 ]; then
|
|
log_error "请使用 sudo 运行此脚本"
|
|
echo "示例: sudo $0"
|
|
exit 1
|
|
fi
|
|
}
|
|
|
|
# 检查并关闭 Cursor 进程(保存进程信息)
|
|
check_and_kill_cursor() {
|
|
log_info "🔍 [检查] 检查 Cursor 进程..."
|
|
|
|
local attempt=1
|
|
local max_attempts=5
|
|
|
|
# 💾 保存Cursor进程路径
|
|
CURSOR_PROCESS_PATH="/Applications/Cursor.app/Contents/MacOS/Cursor"
|
|
|
|
# 函数:获取进程详细信息
|
|
get_process_details() {
|
|
local process_name="$1"
|
|
log_debug "正在获取 $process_name 进程详细信息:"
|
|
ps aux | grep -i "/Applications/Cursor.app" | grep -v grep
|
|
}
|
|
|
|
while [ $attempt -le $max_attempts ]; do
|
|
# 使用更精确的匹配来获取 Cursor 进程
|
|
CURSOR_PIDS=$(ps aux | grep -i "/Applications/Cursor.app" | grep -v grep | awk '{print $2}')
|
|
|
|
if [ -z "$CURSOR_PIDS" ]; then
|
|
log_info "💡 [提示] 未发现运行中的 Cursor 进程"
|
|
# 确认Cursor应用路径存在
|
|
if [ -f "$CURSOR_PROCESS_PATH" ]; then
|
|
log_info "💾 [保存] 已保存Cursor路径: $CURSOR_PROCESS_PATH"
|
|
else
|
|
log_warn "⚠️ [警告] 未找到Cursor应用,请确认已安装"
|
|
fi
|
|
return 0
|
|
fi
|
|
|
|
log_warn "⚠️ [警告] 发现 Cursor 进程正在运行"
|
|
# 💾 保存进程信息
|
|
log_info "💾 [保存] 已保存Cursor路径: $CURSOR_PROCESS_PATH"
|
|
get_process_details "cursor"
|
|
|
|
log_warn "🔄 [操作] 尝试关闭 Cursor 进程..."
|
|
|
|
if [ $attempt -eq $max_attempts ]; then
|
|
log_warn "💥 [强制] 尝试强制终止进程..."
|
|
kill -9 $CURSOR_PIDS 2>/dev/null || true
|
|
else
|
|
kill $CURSOR_PIDS 2>/dev/null || true
|
|
fi
|
|
|
|
sleep 3
|
|
|
|
# 同样使用更精确的匹配来检查进程是否还在运行
|
|
if ! ps aux | grep -i "/Applications/Cursor.app" | grep -v grep > /dev/null; then
|
|
log_info "✅ [成功] Cursor 进程已成功关闭"
|
|
return 0
|
|
fi
|
|
|
|
log_warn "⏳ [等待] 等待进程关闭,尝试 $attempt/$max_attempts..."
|
|
((attempt++))
|
|
done
|
|
|
|
log_error "❌ [错误] 在 $max_attempts 次尝试后仍无法关闭 Cursor 进程"
|
|
get_process_details "cursor"
|
|
log_error "💥 [错误] 请手动关闭进程后重试"
|
|
exit 1
|
|
}
|
|
|
|
# 备份配置文件
|
|
backup_config() {
|
|
if [ ! -f "$STORAGE_FILE" ]; then
|
|
log_warn "配置文件不存在,跳过备份"
|
|
return 0
|
|
fi
|
|
|
|
mkdir -p "$BACKUP_DIR"
|
|
local backup_file="$BACKUP_DIR/storage.json.backup_$(date +%Y%m%d_%H%M%S)"
|
|
|
|
if cp "$STORAGE_FILE" "$backup_file"; then
|
|
chmod 644 "$backup_file"
|
|
chown "$CURRENT_USER" "$backup_file"
|
|
log_info "配置已备份到: $backup_file"
|
|
else
|
|
log_error "备份失败"
|
|
exit 1
|
|
fi
|
|
}
|
|
|
|
# 重新设置配置文件只读(避免权限修复覆盖)
|
|
protect_storage_file() {
|
|
if [ -f "$STORAGE_FILE" ]; then
|
|
if chmod 444 "$STORAGE_FILE" 2>/dev/null; then
|
|
log_info "🔒 [保护] 配置文件已重新设置为只读"
|
|
else
|
|
log_warn "⚠️ [保护] 配置文件只读设置失败"
|
|
fi
|
|
fi
|
|
}
|
|
|
|
# 🔧 修改Cursor内核JS文件实现设备识别绕过(增强版三重方案)
|
|
# 方案A: someValue占位符替换 - 稳定锚点,不依赖混淆后的函数名
|
|
# 方案B: b6 定点重写 - 机器码源函数直接返回固定值
|
|
# 方案C: Loader Stub + 外置 Hook - 主/共享进程仅加载外置 Hook 文件
|
|
modify_cursor_js_files() {
|
|
log_info "🔧 [内核修改] 开始修改Cursor内核JS文件实现设备识别绕过..."
|
|
log_info "💡 [方案] 使用增强版三重方案:占位符替换 + b6 定点重写 + Loader Stub + 外置 Hook"
|
|
echo
|
|
|
|
# 检查Cursor应用是否存在
|
|
if [ ! -d "$CURSOR_APP_PATH" ]; then
|
|
log_error "❌ [错误] 未找到Cursor应用: $CURSOR_APP_PATH"
|
|
return 1
|
|
fi
|
|
|
|
# 生成或复用设备标识符(优先使用配置中生成的值)
|
|
local machine_id="${CURSOR_ID_MACHINE_ID:-}"
|
|
local machine_guid="${CURSOR_ID_MACHINE_GUID:-}"
|
|
local device_id="${CURSOR_ID_DEVICE_ID:-}"
|
|
local mac_machine_id="${CURSOR_ID_MAC_MACHINE_ID:-}"
|
|
local sqm_id="${CURSOR_ID_SQM_ID:-}"
|
|
local session_id="${CURSOR_ID_SESSION_ID:-}"
|
|
local first_session_date="${CURSOR_ID_FIRST_SESSION_DATE:-}"
|
|
local mac_address="${CURSOR_ID_MAC_ADDRESS:-00:11:22:33:44:55}"
|
|
local ids_missing=false
|
|
|
|
if [ -z "$machine_id" ]; then
|
|
machine_id=$(generate_hex_bytes 32)
|
|
ids_missing=true
|
|
fi
|
|
if [ -z "$machine_guid" ]; then
|
|
machine_guid=$(generate_uuid)
|
|
ids_missing=true
|
|
fi
|
|
if [ -z "$device_id" ]; then
|
|
device_id=$(generate_uuid)
|
|
ids_missing=true
|
|
fi
|
|
if [ -z "$mac_machine_id" ]; then
|
|
mac_machine_id=$(generate_hex_bytes 32)
|
|
ids_missing=true
|
|
fi
|
|
if [ -z "$sqm_id" ]; then
|
|
sqm_id="{$(generate_uuid | tr '[:lower:]' '[:upper:]')}"
|
|
ids_missing=true
|
|
fi
|
|
if [ -z "$session_id" ]; then
|
|
session_id=$(generate_uuid)
|
|
ids_missing=true
|
|
fi
|
|
if [ -z "$first_session_date" ]; then
|
|
first_session_date=$(date -u +"%Y-%m-%dT%H:%M:%S.000Z")
|
|
ids_missing=true
|
|
fi
|
|
|
|
if [ "$ids_missing" = true ]; then
|
|
log_warn "部分 ID 未就绪,已生成新值用于 JS 注入"
|
|
else
|
|
log_info "已使用配置中的设备标识符进行 JS 注入"
|
|
fi
|
|
|
|
CURSOR_ID_MACHINE_ID="$machine_id"
|
|
CURSOR_ID_MACHINE_GUID="$machine_guid"
|
|
CURSOR_ID_DEVICE_ID="$device_id"
|
|
CURSOR_ID_MAC_MACHINE_ID="$mac_machine_id"
|
|
CURSOR_ID_SQM_ID="$sqm_id"
|
|
CURSOR_ID_SESSION_ID="$session_id"
|
|
CURSOR_ID_FIRST_SESSION_DATE="$first_session_date"
|
|
CURSOR_ID_MAC_ADDRESS="$mac_address"
|
|
|
|
log_info "🔑 [准备] 设备标识符已就绪"
|
|
log_info " machineId: ${machine_id:0:16}..."
|
|
log_info " machineGuid: ${machine_guid:0:16}..."
|
|
log_info " deviceId: ${device_id:0:16}..."
|
|
log_info " macMachineId: ${mac_machine_id:0:16}..."
|
|
log_info " sqmId: $sqm_id"
|
|
|
|
# 保存 ID 配置到用户目录(供 Hook 读取)
|
|
# 每次执行都删除旧配置并重新生成,确保获得新的设备标识符
|
|
local ids_config_path="$TARGET_HOME/.cursor_ids.json"
|
|
if [ -f "$ids_config_path" ]; then
|
|
rm -f "$ids_config_path"
|
|
log_info "🗑️ [清理] 已删除旧的 ID 配置文件"
|
|
fi
|
|
cat > "$ids_config_path" << EOF
|
|
{
|
|
"machineId": "$machine_id",
|
|
"machineGuid": "$machine_guid",
|
|
"macMachineId": "$mac_machine_id",
|
|
"devDeviceId": "$device_id",
|
|
"sqmId": "$sqm_id",
|
|
"macAddress": "$mac_address",
|
|
"sessionId": "$session_id",
|
|
"firstSessionDate": "$first_session_date",
|
|
"createdAt": "$first_session_date"
|
|
}
|
|
EOF
|
|
log_info "💾 [保存] 新的 ID 配置已保存到: $ids_config_path"
|
|
|
|
# 部署外置 Hook 文件(供 Loader Stub 加载,支持多域名备用下载)
|
|
local hook_target_path="$TARGET_HOME/.cursor_hook.js"
|
|
local script_dir
|
|
script_dir="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
|
|
local hook_source_path="$script_dir/../hook/cursor_hook.js"
|
|
local hook_download_urls=(
|
|
"https://wget.la/https://raw.githubusercontent.com/yuaotian/go-cursor-help/refs/heads/master/scripts/hook/cursor_hook.js"
|
|
"https://down.npee.cn/?https://raw.githubusercontent.com/yuaotian/go-cursor-help/refs/heads/master/scripts/hook/cursor_hook.js"
|
|
"https://xget.xi-xu.me/gh/yuaotian/go-cursor-help/refs/heads/master/scripts/hook/cursor_hook.js"
|
|
"https://gh-proxy.com/https://raw.githubusercontent.com/yuaotian/go-cursor-help/refs/heads/master/scripts/hook/cursor_hook.js"
|
|
"https://gh.chjina.com/https://raw.githubusercontent.com/yuaotian/go-cursor-help/refs/heads/master/scripts/hook/cursor_hook.js"
|
|
)
|
|
# 支持通过环境变量覆盖下载节点(逗号分隔)
|
|
if [ -n "$CURSOR_HOOK_DOWNLOAD_URLS" ]; then
|
|
IFS=',' read -r -a hook_download_urls <<< "$CURSOR_HOOK_DOWNLOAD_URLS"
|
|
log_info "ℹ️ [Hook] 检测到自定义下载节点列表,将优先使用"
|
|
fi
|
|
|
|
if [ -f "$hook_source_path" ]; then
|
|
if cp "$hook_source_path" "$hook_target_path"; then
|
|
chown "$TARGET_USER":"$(id -g -n "$TARGET_USER")" "$hook_target_path" 2>/dev/null || true
|
|
log_info "✅ [Hook] 外置 Hook 已部署: $hook_target_path"
|
|
else
|
|
log_warn "⚠️ [Hook] 本地 Hook 复制失败,尝试在线下载..."
|
|
fi
|
|
fi
|
|
|
|
if [ ! -f "$hook_target_path" ]; then
|
|
log_info "ℹ️ [Hook] 正在下载外置 Hook,用于设备标识拦截..."
|
|
local hook_downloaded=false
|
|
local total_urls=${#hook_download_urls[@]}
|
|
if [ "$total_urls" -eq 0 ]; then
|
|
log_warn "⚠️ [Hook] 下载节点列表为空,跳过在线下载"
|
|
elif command -v curl >/dev/null 2>&1; then
|
|
local index=0
|
|
for url in "${hook_download_urls[@]}"; do
|
|
index=$((index + 1))
|
|
log_info "⏳ [Hook] ($index/$total_urls) 当前下载节点: $url"
|
|
if curl -fL --progress-bar "$url" -o "$hook_target_path"; then
|
|
chown "$TARGET_USER":"$(id -g -n "$TARGET_USER")" "$hook_target_path" 2>/dev/null || true
|
|
log_info "✅ [Hook] 外置 Hook 已在线下载: $hook_target_path"
|
|
hook_downloaded=true
|
|
break
|
|
else
|
|
rm -f "$hook_target_path"
|
|
log_warn "⚠️ [Hook] 外置 Hook 下载失败: $url"
|
|
fi
|
|
done
|
|
elif command -v wget >/dev/null 2>&1; then
|
|
local index=0
|
|
for url in "${hook_download_urls[@]}"; do
|
|
index=$((index + 1))
|
|
log_info "⏳ [Hook] ($index/$total_urls) 当前下载节点: $url"
|
|
if wget --progress=bar:force -O "$hook_target_path" "$url"; then
|
|
chown "$TARGET_USER":"$(id -g -n "$TARGET_USER")" "$hook_target_path" 2>/dev/null || true
|
|
log_info "✅ [Hook] 外置 Hook 已在线下载: $hook_target_path"
|
|
hook_downloaded=true
|
|
break
|
|
else
|
|
rm -f "$hook_target_path"
|
|
log_warn "⚠️ [Hook] 外置 Hook 下载失败: $url"
|
|
fi
|
|
done
|
|
else
|
|
log_warn "⚠️ [Hook] 未检测到 curl/wget,无法在线下载 Hook"
|
|
fi
|
|
if [ "$hook_downloaded" != true ] && [ ! -f "$hook_target_path" ]; then
|
|
log_warn "⚠️ [Hook] 外置 Hook 全部下载失败"
|
|
fi
|
|
fi
|
|
|
|
# 目标JS文件列表(main + shared process)
|
|
local js_files=(
|
|
"$CURSOR_APP_PATH/Contents/Resources/app/out/main.js"
|
|
"$CURSOR_APP_PATH/Contents/Resources/app/out/vs/code/electron-utility/sharedProcess/sharedProcessMain.js"
|
|
)
|
|
|
|
local modified_count=0
|
|
|
|
# 关闭Cursor进程
|
|
log_info "🔄 [关闭] 关闭Cursor进程以进行文件修改..."
|
|
check_and_kill_cursor
|
|
|
|
# 创建备份目录
|
|
local timestamp=$(date +%Y%m%d_%H%M%S)
|
|
local backup_dir="$CURSOR_APP_PATH/Contents/Resources/app/out/backups"
|
|
|
|
log_info "💾 [备份] 创建JS文件备份..."
|
|
mkdir -p "$backup_dir"
|
|
|
|
# 处理每个文件:创建原始备份或从原始备份恢复
|
|
for file in "${js_files[@]}"; do
|
|
if [ ! -f "$file" ]; then
|
|
log_warn "⚠️ [警告] 文件不存在: ${file/$CURSOR_APP_PATH\//}"
|
|
continue
|
|
fi
|
|
|
|
local file_name=$(basename "$file")
|
|
local file_original_backup="$backup_dir/$file_name.original"
|
|
|
|
# 如果原始备份不存在,先创建
|
|
if [ ! -f "$file_original_backup" ]; then
|
|
# 检查当前文件是否已被修改过
|
|
if grep -q "__cursor_patched__" "$file" 2>/dev/null; then
|
|
log_warn "⚠️ [警告] 文件已被修改但无原始备份,将使用当前版本作为基础"
|
|
fi
|
|
cp "$file" "$file_original_backup"
|
|
log_info "✅ [备份] 原始备份创建成功: $file_name"
|
|
else
|
|
# 从原始备份恢复,确保每次都是干净的注入
|
|
log_info "🔄 [恢复] 从原始备份恢复: $file_name"
|
|
cp "$file_original_backup" "$file"
|
|
fi
|
|
done
|
|
|
|
# 创建时间戳备份(记录每次修改前的状态)
|
|
for file in "${js_files[@]}"; do
|
|
if [ -f "$file" ]; then
|
|
cp "$file" "$backup_dir/$(basename "$file").backup_$timestamp"
|
|
fi
|
|
done
|
|
log_info "✅ [备份] 时间戳备份创建成功: $backup_dir"
|
|
|
|
# 修改JS文件(每次都重新注入,因为已从原始备份恢复)
|
|
log_info "🔧 [修改] 开始修改JS文件(使用新的设备标识符)..."
|
|
|
|
for file in "${js_files[@]}"; do
|
|
if [ ! -f "$file" ]; then
|
|
log_warn "⚠️ [跳过] 文件不存在: ${file/$CURSOR_APP_PATH\//}"
|
|
continue
|
|
fi
|
|
|
|
log_info "📝 [处理] 正在处理: ${file/$CURSOR_APP_PATH\//}"
|
|
|
|
# ========== 方法A: someValue占位符替换(稳定锚点) ==========
|
|
# 重要说明:
|
|
# 当前 Cursor 的 main.js 中占位符通常是以字符串字面量形式出现,例如:
|
|
# this.machineId="someValue.machineId"
|
|
# 如果直接把 someValue.machineId 替换成 "\"<真实值>\"",会形成 ""<真实值>"" 导致 JS 语法错误。
|
|
# 因此这里优先替换完整的字符串字面量(包含外层引号),再兜底替换不带引号的占位符。
|
|
local replaced=false
|
|
|
|
if grep -q 'someValue\.machineId' "$file"; then
|
|
sed -i.tmp \
|
|
-e "s/\"someValue\.machineId\"/\"${machine_id}\"/g" \
|
|
-e "s/'someValue\.machineId'/\"${machine_id}\"/g" \
|
|
-e "s/someValue\.machineId/\"${machine_id}\"/g" \
|
|
"$file"
|
|
log_info " ✓ [方案A] 替换 someValue.machineId"
|
|
replaced=true
|
|
fi
|
|
|
|
if grep -q 'someValue\.macMachineId' "$file"; then
|
|
sed -i.tmp \
|
|
-e "s/\"someValue\.macMachineId\"/\"${mac_machine_id}\"/g" \
|
|
-e "s/'someValue\.macMachineId'/\"${mac_machine_id}\"/g" \
|
|
-e "s/someValue\.macMachineId/\"${mac_machine_id}\"/g" \
|
|
"$file"
|
|
log_info " ✓ [方案A] 替换 someValue.macMachineId"
|
|
replaced=true
|
|
fi
|
|
|
|
if grep -q 'someValue\.devDeviceId' "$file"; then
|
|
sed -i.tmp \
|
|
-e "s/\"someValue\.devDeviceId\"/\"${device_id}\"/g" \
|
|
-e "s/'someValue\.devDeviceId'/\"${device_id}\"/g" \
|
|
-e "s/someValue\.devDeviceId/\"${device_id}\"/g" \
|
|
"$file"
|
|
log_info " ✓ [方案A] 替换 someValue.devDeviceId"
|
|
replaced=true
|
|
fi
|
|
|
|
if grep -q 'someValue\.sqmId' "$file"; then
|
|
sed -i.tmp \
|
|
-e "s/\"someValue\.sqmId\"/\"${sqm_id}\"/g" \
|
|
-e "s/'someValue\.sqmId'/\"${sqm_id}\"/g" \
|
|
-e "s/someValue\.sqmId/\"${sqm_id}\"/g" \
|
|
"$file"
|
|
log_info " ✓ [方案A] 替换 someValue.sqmId"
|
|
replaced=true
|
|
fi
|
|
|
|
if grep -q 'someValue\.sessionId' "$file"; then
|
|
sed -i.tmp \
|
|
-e "s/\"someValue\.sessionId\"/\"${session_id}\"/g" \
|
|
-e "s/'someValue\.sessionId'/\"${session_id}\"/g" \
|
|
-e "s/someValue\.sessionId/\"${session_id}\"/g" \
|
|
"$file"
|
|
log_info " ✓ [方案A] 替换 someValue.sessionId"
|
|
replaced=true
|
|
fi
|
|
|
|
if grep -q 'someValue\.firstSessionDate' "$file"; then
|
|
sed -i.tmp \
|
|
-e "s/\"someValue\.firstSessionDate\"/\"${first_session_date}\"/g" \
|
|
-e "s/'someValue\.firstSessionDate'/\"${first_session_date}\"/g" \
|
|
-e "s/someValue\.firstSessionDate/\"${first_session_date}\"/g" \
|
|
"$file"
|
|
log_info " ✓ [方案A] 替换 someValue.firstSessionDate"
|
|
replaced=true
|
|
fi
|
|
|
|
# ========== 方法B: b6 定点重写(机器码源函数,仅 main.js) ==========
|
|
local b6_patched=false
|
|
if [ "$(basename "$file")" = "main.js" ]; then
|
|
if command -v python3 >/dev/null 2>&1; then
|
|
local b6_result
|
|
b6_result=$(python3 - "$file" "$machine_guid" "$machine_id" <<'PY'
|
|
if True:
|
|
import re, sys
|
|
|
|
def diag(msg):
|
|
print(f"[方案B][诊断] {msg}", file=sys.stderr)
|
|
|
|
path, machine_guid, machine_id = sys.argv[1], sys.argv[2], sys.argv[3]
|
|
|
|
with open(path, "r", encoding="utf-8") as f:
|
|
data = f.read()
|
|
|
|
# ✅ 1+3 融合:限定 out-build/vs/base/node/id.js 模块内做特征匹配 + 花括号配对定位函数边界
|
|
marker = "out-build/vs/base/node/id.js"
|
|
marker_index = data.find(marker)
|
|
if marker_index < 0:
|
|
print("NOT_FOUND")
|
|
diag(f"未找到模块标记: {marker}")
|
|
raise SystemExit(0)
|
|
|
|
window_end = min(len(data), marker_index + 200000)
|
|
window = data[marker_index:window_end]
|
|
|
|
def find_matching_brace(text, open_index, max_scan=20000):
|
|
limit = min(len(text), open_index + max_scan)
|
|
depth = 1
|
|
in_single = in_double = in_template = False
|
|
in_line_comment = in_block_comment = False
|
|
escape = False
|
|
i = open_index + 1
|
|
while i < limit:
|
|
ch = text[i]
|
|
nxt = text[i + 1] if i + 1 < limit else ""
|
|
|
|
if in_line_comment:
|
|
if ch == "\n":
|
|
in_line_comment = False
|
|
i += 1
|
|
continue
|
|
if in_block_comment:
|
|
if ch == "*" and nxt == "/":
|
|
in_block_comment = False
|
|
i += 2
|
|
continue
|
|
i += 1
|
|
continue
|
|
|
|
if in_single:
|
|
if escape:
|
|
escape = False
|
|
elif ch == "\\\\":
|
|
escape = True
|
|
elif ch == "'":
|
|
in_single = False
|
|
i += 1
|
|
continue
|
|
if in_double:
|
|
if escape:
|
|
escape = False
|
|
elif ch == "\\\\":
|
|
escape = True
|
|
elif ch == '"':
|
|
in_double = False
|
|
i += 1
|
|
continue
|
|
if in_template:
|
|
if escape:
|
|
escape = False
|
|
elif ch == "\\\\":
|
|
escape = True
|
|
elif ch == "`":
|
|
in_template = False
|
|
i += 1
|
|
continue
|
|
|
|
if ch == "/" and nxt == "/":
|
|
in_line_comment = True
|
|
i += 2
|
|
continue
|
|
if ch == "/" and nxt == "*":
|
|
in_block_comment = True
|
|
i += 2
|
|
continue
|
|
|
|
if ch == "'":
|
|
in_single = True
|
|
i += 1
|
|
continue
|
|
if ch == '"':
|
|
in_double = True
|
|
i += 1
|
|
continue
|
|
if ch == "`":
|
|
in_template = True
|
|
i += 1
|
|
continue
|
|
|
|
if ch == "{":
|
|
depth += 1
|
|
elif ch == "}":
|
|
depth -= 1
|
|
if depth == 0:
|
|
return i
|
|
|
|
i += 1
|
|
return None
|
|
|
|
# 🔧 修复:避免 raw string + 单引号 + ['"] 字符组导致的语法错误;同时修正正则转义,提升 b6 特征匹配命中率
|
|
hash_re = re.compile(r"""createHash\(["']sha256["']\)""")
|
|
sig_re = re.compile(r'^async function (\w+)\((\w+)\)')
|
|
|
|
hash_matches = list(hash_re.finditer(window))
|
|
diag(f"marker_index={marker_index} window_len={len(window)} sha256_createHash={len(hash_matches)}")
|
|
|
|
for idx, hm in enumerate(hash_matches, start=1):
|
|
hash_pos = hm.start()
|
|
func_start = window.rfind("async function", 0, hash_pos)
|
|
if func_start < 0:
|
|
if idx <= 3:
|
|
diag(f"候选#{idx}: 未找到 async function 起点")
|
|
continue
|
|
|
|
open_brace = window.find("{", func_start)
|
|
if open_brace < 0:
|
|
if idx <= 3:
|
|
diag(f"候选#{idx}: 未找到函数起始花括号")
|
|
continue
|
|
|
|
end_brace = find_matching_brace(window, open_brace, max_scan=20000)
|
|
if end_brace is None:
|
|
if idx <= 3:
|
|
diag(f"候选#{idx}: 花括号配对失败(扫描上限内未闭合)")
|
|
continue
|
|
|
|
func_text = window[func_start:end_brace + 1]
|
|
if len(func_text) > 8000:
|
|
if idx <= 3:
|
|
diag(f"候选#{idx}: 函数体过长 len={len(func_text)},已跳过")
|
|
continue
|
|
|
|
sm = sig_re.match(func_text)
|
|
if not sm:
|
|
if idx <= 3:
|
|
diag(f"候选#{idx}: 未解析到函数签名(async function name(param))")
|
|
continue
|
|
name, param = sm.group(1), sm.group(2)
|
|
|
|
# 特征校验:sha256 + hex digest + return param ? raw : hash
|
|
has_digest = re.search(r"""\.digest\(["']hex["']\)""", func_text) is not None
|
|
has_return = re.search(r'return\s+' + re.escape(param) + r'\?\w+:\w+\}', func_text) is not None
|
|
if idx <= 3:
|
|
diag(f"候选#{idx}: {name}({param}) len={len(func_text)} digest={has_digest} return={has_return}")
|
|
if not has_digest:
|
|
continue
|
|
if not has_return:
|
|
continue
|
|
|
|
replacement = f'async function {name}({param}){{return {param}?"{machine_guid}":"{machine_id}";}}'
|
|
abs_start = marker_index + func_start
|
|
abs_end = marker_index + end_brace
|
|
new_data = data[:abs_start] + replacement + data[abs_end + 1:]
|
|
with open(path, "w", encoding="utf-8") as f:
|
|
f.write(new_data)
|
|
diag(f"命中并重写: {name}({param}) len={len(func_text)}")
|
|
print("PATCHED")
|
|
break
|
|
else:
|
|
diag("未找到满足特征的候选函数")
|
|
print("NOT_FOUND")
|
|
PY
|
|
)
|
|
if [ "$b6_result" = "PATCHED" ]; then
|
|
log_info " ✓ [方案B] 已重写 b6 特征函数"
|
|
b6_patched=true
|
|
else
|
|
log_warn "⚠️ [方案B] 未定位到 b6 特征函数"
|
|
fi
|
|
else
|
|
log_warn "⚠️ [方案B] 未检测到 python3,跳过 b6 定点重写"
|
|
fi
|
|
fi
|
|
|
|
# ========== 方法C: Loader Stub 注入 ==========
|
|
local inject_code='// ========== Cursor Hook Loader 开始 ==========
|
|
;(async function(){/*__cursor_patched__*/
|
|
"use strict";
|
|
if(globalThis.__cursor_hook_loaded__)return;
|
|
globalThis.__cursor_hook_loaded__=true;
|
|
|
|
try{
|
|
// 兼容 ESM/CJS:避免使用 import.meta(仅 ESM 支持),统一用动态 import 加载 Hook
|
|
var fsMod=await import("fs");
|
|
var pathMod=await import("path");
|
|
var osMod=await import("os");
|
|
var urlMod=await import("url");
|
|
|
|
var fs=fsMod&&(fsMod.default||fsMod);
|
|
var path=pathMod&&(pathMod.default||pathMod);
|
|
var os=osMod&&(osMod.default||osMod);
|
|
var url=urlMod&&(urlMod.default||urlMod);
|
|
|
|
if(fs&&path&&os&&url&&typeof url.pathToFileURL==="function"){
|
|
var hookPath=path.join(os.homedir(), ".cursor_hook.js");
|
|
if(typeof fs.existsSync==="function"&&fs.existsSync(hookPath)){
|
|
await import(url.pathToFileURL(hookPath).href);
|
|
}
|
|
}
|
|
}catch(e){
|
|
// 失败静默,避免影响启动
|
|
}
|
|
})();
|
|
// ========== Cursor Hook Loader 结束 ==========
|
|
|
|
'
|
|
|
|
# 在版权声明后注入代码
|
|
if grep -q '\*/' "$file"; then
|
|
# 使用 awk 在版权声明后注入
|
|
awk -v inject="$inject_code" '
|
|
/\*\// && !injected {
|
|
print
|
|
print ""
|
|
print inject
|
|
injected = 1
|
|
next
|
|
}
|
|
{ print }
|
|
' "$file" > "${file}.new"
|
|
mv "${file}.new" "$file"
|
|
log_info " ✓ [方案C] Loader Stub 已注入(版权声明后)"
|
|
else
|
|
# 注入到文件开头
|
|
echo "$inject_code" > "${file}.new"
|
|
cat "$file" >> "${file}.new"
|
|
mv "${file}.new" "$file"
|
|
log_info " ✓ [方案C] Loader Stub 已注入(文件开头)"
|
|
fi
|
|
|
|
# 清理临时文件
|
|
rm -f "${file}.tmp"
|
|
|
|
local summary="Hook加载器"
|
|
if [ "$replaced" = true ]; then
|
|
summary="someValue替换 + $summary"
|
|
fi
|
|
if [ "$b6_patched" = true ]; then
|
|
summary="b6定点重写 + $summary"
|
|
fi
|
|
log_info "✅ [成功] 增强版方案修改成功($summary)"
|
|
((modified_count++))
|
|
done
|
|
|
|
if [ $modified_count -gt 0 ]; then
|
|
log_info "🎉 [完成] 成功修改 $modified_count 个JS文件"
|
|
log_info "💾 [备份] 原始文件备份位置: $backup_dir"
|
|
log_info "💡 [说明] 使用增强版三重方案:"
|
|
log_info " • 方案A: someValue占位符替换(稳定锚点,跨版本兼容)"
|
|
log_info " • 方案B: b6 定点重写(机器码源函数)"
|
|
log_info " • 方案C: Loader Stub + 外置 Hook(cursor_hook.js)"
|
|
log_info "📁 [配置] ID 配置文件: $ids_config_path"
|
|
return 0
|
|
else
|
|
log_error "❌ [失败] 没有成功修改任何文件"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
|
|
|
|
|
|
# 修改现有文件
|
|
modify_or_add_config() {
|
|
local key="$1"
|
|
local value="$2"
|
|
local file="$3"
|
|
|
|
if [ ! -f "$file" ]; then
|
|
log_error "文件不存在: $file"
|
|
return 1
|
|
fi
|
|
|
|
# 确保文件可写
|
|
chmod 644 "$file" || {
|
|
log_error "无法修改文件权限: $file"
|
|
return 1
|
|
}
|
|
|
|
# 创建临时文件
|
|
local temp_file=$(mktemp)
|
|
|
|
# 检查key是否存在
|
|
if grep -q "\"$key\":" "$file"; then
|
|
# key存在,执行替换
|
|
sed "s/\"$key\":[[:space:]]*\"[^\"]*\"/\"$key\": \"$value\"/" "$file" > "$temp_file" || {
|
|
log_error "修改配置失败: $key"
|
|
rm -f "$temp_file"
|
|
return 1
|
|
}
|
|
else
|
|
# key不存在,添加新的key-value对
|
|
sed "s/}$/,\n \"$key\": \"$value\"\n}/" "$file" > "$temp_file" || {
|
|
log_error "添加配置失败: $key"
|
|
rm -f "$temp_file"
|
|
return 1
|
|
}
|
|
fi
|
|
|
|
# 检查临时文件是否为空
|
|
if [ ! -s "$temp_file" ]; then
|
|
log_error "生成的临时文件为空"
|
|
rm -f "$temp_file"
|
|
return 1
|
|
fi
|
|
|
|
# 使用 cat 替换原文件内容
|
|
cat "$temp_file" > "$file" || {
|
|
log_error "无法写入文件: $file"
|
|
rm -f "$temp_file"
|
|
return 1
|
|
}
|
|
|
|
rm -f "$temp_file"
|
|
|
|
# 恢复文件权限
|
|
chmod 444 "$file"
|
|
|
|
return 0
|
|
}
|
|
|
|
# 清理 Cursor 之前的修改
|
|
clean_cursor_app() {
|
|
log_info "尝试清理 Cursor 之前的修改..."
|
|
|
|
# 如果存在备份,直接恢复备份
|
|
local latest_backup=""
|
|
|
|
# 查找最新的备份
|
|
latest_backup=$(find /tmp -name "Cursor.app.backup_*" -type d -print 2>/dev/null | sort -r | head -1)
|
|
|
|
if [ -n "$latest_backup" ] && [ -d "$latest_backup" ]; then
|
|
log_info "找到现有备份: $latest_backup"
|
|
log_info "正在恢复原始版本..."
|
|
|
|
# 停止 Cursor 进程
|
|
check_and_kill_cursor
|
|
|
|
# 恢复备份
|
|
sudo rm -rf "$CURSOR_APP_PATH"
|
|
sudo cp -R "$latest_backup" "$CURSOR_APP_PATH"
|
|
sudo chown -R "$CURRENT_USER:staff" "$CURSOR_APP_PATH"
|
|
sudo chmod -R 755 "$CURSOR_APP_PATH"
|
|
|
|
log_info "已恢复原始版本"
|
|
return 0
|
|
else
|
|
log_warn "未找到现有备份,尝试重新安装 Cursor..."
|
|
echo "您可以从 https://cursor.sh 下载并重新安装 Cursor"
|
|
echo "或者继续执行此脚本,将尝试修复现有安装"
|
|
|
|
# 可以在这里添加重新下载和安装的逻辑
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 修改 Cursor 主程序文件(安全模式)
|
|
modify_cursor_app_files() {
|
|
log_info "正在安全修改 Cursor 主程序文件..."
|
|
log_info "详细日志将记录到: $LOG_FILE"
|
|
|
|
# 先清理之前的修改
|
|
clean_cursor_app
|
|
|
|
# 验证应用是否存在
|
|
if [ ! -d "$CURSOR_APP_PATH" ]; then
|
|
log_error "未找到 Cursor.app,请确认安装路径: $CURSOR_APP_PATH"
|
|
return 1
|
|
fi
|
|
|
|
# 定义目标文件 - 将extensionHostProcess.js放在最前面优先处理
|
|
local target_files=(
|
|
"${CURSOR_APP_PATH}/Contents/Resources/app/out/vs/workbench/api/node/extensionHostProcess.js"
|
|
"${CURSOR_APP_PATH}/Contents/Resources/app/out/main.js"
|
|
"${CURSOR_APP_PATH}/Contents/Resources/app/out/vs/code/node/cliProcessMain.js"
|
|
)
|
|
|
|
# 检查文件是否存在并且是否已修改
|
|
local need_modification=false
|
|
local missing_files=false
|
|
|
|
log_debug "检查目标文件..."
|
|
for file in "${target_files[@]}"; do
|
|
if [ ! -f "$file" ]; then
|
|
log_warn "文件不存在: ${file/$CURSOR_APP_PATH\//}"
|
|
echo "[FILE_CHECK] 文件不存在: $file" >> "$LOG_FILE"
|
|
missing_files=true
|
|
continue
|
|
fi
|
|
|
|
echo "[FILE_CHECK] 文件存在: $file ($(wc -c < "$file") 字节)" >> "$LOG_FILE"
|
|
|
|
if ! grep -q "return crypto.randomUUID()" "$file" 2>/dev/null; then
|
|
log_info "文件需要修改: ${file/$CURSOR_APP_PATH\//}"
|
|
grep -n "IOPlatformUUID" "$file" | head -3 >> "$LOG_FILE" || echo "[FILE_CHECK] 未找到 IOPlatformUUID" >> "$LOG_FILE"
|
|
need_modification=true
|
|
break
|
|
else
|
|
log_info "文件已修改: ${file/$CURSOR_APP_PATH\//}"
|
|
fi
|
|
done
|
|
|
|
# 如果所有文件都已修改或不存在,则退出
|
|
if [ "$missing_files" = true ]; then
|
|
log_error "部分目标文件不存在,请确认 Cursor 安装是否完整"
|
|
return 1
|
|
fi
|
|
|
|
if [ "$need_modification" = false ]; then
|
|
log_info "所有目标文件已经被修改过,无需重复操作"
|
|
return 0
|
|
fi
|
|
|
|
# 创建临时工作目录
|
|
local timestamp=$(date +%Y%m%d_%H%M%S)
|
|
local temp_dir="/tmp/cursor_reset_${timestamp}"
|
|
local temp_app="${temp_dir}/Cursor.app"
|
|
local backup_app="/tmp/Cursor.app.backup_${timestamp}"
|
|
|
|
log_debug "创建临时目录: $temp_dir"
|
|
echo "[TEMP_DIR] 创建临时目录: $temp_dir" >> "$LOG_FILE"
|
|
|
|
# 清理可能存在的旧临时目录
|
|
if [ -d "$temp_dir" ]; then
|
|
log_info "清理已存在的临时目录..."
|
|
rm -rf "$temp_dir"
|
|
fi
|
|
|
|
# 创建新的临时目录
|
|
mkdir -p "$temp_dir" || {
|
|
log_error "无法创建临时目录: $temp_dir"
|
|
echo "[ERROR] 无法创建临时目录: $temp_dir" >> "$LOG_FILE"
|
|
return 1
|
|
}
|
|
|
|
# 备份原应用
|
|
log_info "备份原应用..."
|
|
echo "[BACKUP] 开始备份: $CURSOR_APP_PATH -> $backup_app" >> "$LOG_FILE"
|
|
|
|
cp -R "$CURSOR_APP_PATH" "$backup_app" || {
|
|
log_error "无法创建应用备份"
|
|
echo "[ERROR] 备份失败: $CURSOR_APP_PATH -> $backup_app" >> "$LOG_FILE"
|
|
rm -rf "$temp_dir"
|
|
return 1
|
|
}
|
|
|
|
echo "[BACKUP] 备份完成" >> "$LOG_FILE"
|
|
|
|
# 复制应用到临时目录
|
|
log_info "创建临时工作副本..."
|
|
echo "[COPY] 开始复制: $CURSOR_APP_PATH -> $temp_dir" >> "$LOG_FILE"
|
|
|
|
cp -R "$CURSOR_APP_PATH" "$temp_dir" || {
|
|
log_error "无法复制应用到临时目录"
|
|
echo "[ERROR] 复制失败: $CURSOR_APP_PATH -> $temp_dir" >> "$LOG_FILE"
|
|
rm -rf "$temp_dir" "$backup_app"
|
|
return 1
|
|
}
|
|
|
|
echo "[COPY] 复制完成" >> "$LOG_FILE"
|
|
|
|
# 确保临时目录的权限正确
|
|
chown -R "$CURRENT_USER:staff" "$temp_dir"
|
|
chmod -R 755 "$temp_dir"
|
|
|
|
# 移除签名(增强兼容性)
|
|
log_info "移除应用签名..."
|
|
echo "[CODESIGN] 移除签名: $temp_app" >> "$LOG_FILE"
|
|
|
|
codesign --remove-signature "$temp_app" 2>> "$LOG_FILE" || {
|
|
log_warn "移除应用签名失败"
|
|
echo "[WARN] 移除签名失败: $temp_app" >> "$LOG_FILE"
|
|
}
|
|
|
|
# 移除所有相关组件的签名
|
|
local components=(
|
|
"$temp_app/Contents/Frameworks/Cursor Helper.app"
|
|
"$temp_app/Contents/Frameworks/Cursor Helper (GPU).app"
|
|
"$temp_app/Contents/Frameworks/Cursor Helper (Plugin).app"
|
|
"$temp_app/Contents/Frameworks/Cursor Helper (Renderer).app"
|
|
)
|
|
|
|
for component in "${components[@]}"; do
|
|
if [ -e "$component" ]; then
|
|
log_info "正在移除签名: $component"
|
|
codesign --remove-signature "$component" || {
|
|
log_warn "移除组件签名失败: $component"
|
|
}
|
|
fi
|
|
done
|
|
|
|
# 修改目标文件 - 优先处理js文件
|
|
local modified_count=0
|
|
local files=(
|
|
"${temp_app}/Contents/Resources/app/out/vs/workbench/api/node/extensionHostProcess.js"
|
|
"${temp_app}/Contents/Resources/app/out/main.js"
|
|
"${temp_app}/Contents/Resources/app/out/vs/code/node/cliProcessMain.js"
|
|
)
|
|
|
|
for file in "${files[@]}"; do
|
|
if [ ! -f "$file" ]; then
|
|
log_warn "文件不存在: ${file/$temp_dir\//}"
|
|
continue
|
|
fi
|
|
|
|
log_debug "处理文件: ${file/$temp_dir\//}"
|
|
echo "[PROCESS] 开始处理文件: $file" >> "$LOG_FILE"
|
|
echo "[PROCESS] 文件大小: $(wc -c < "$file") 字节" >> "$LOG_FILE"
|
|
|
|
# 输出文件部分内容到日志
|
|
echo "[FILE_CONTENT] 文件头部 100 行:" >> "$LOG_FILE"
|
|
head -100 "$file" 2>/dev/null | grep -v "^$" | head -50 >> "$LOG_FILE"
|
|
echo "[FILE_CONTENT] ..." >> "$LOG_FILE"
|
|
|
|
# 创建文件备份
|
|
cp "$file" "${file}.bak" || {
|
|
log_error "无法创建文件备份: ${file/$temp_dir\//}"
|
|
echo "[ERROR] 无法创建文件备份: $file" >> "$LOG_FILE"
|
|
continue
|
|
}
|
|
|
|
# 使用 sed 替换而不是字符串操作
|
|
if [[ "$file" == *"extensionHostProcess.js"* ]]; then
|
|
log_debug "处理 extensionHostProcess.js 文件..."
|
|
echo "[PROCESS_DETAIL] 开始处理 extensionHostProcess.js 文件" >> "$LOG_FILE"
|
|
|
|
# 检查是否包含目标代码
|
|
if grep -q 'i.header.set("x-cursor-checksum' "$file"; then
|
|
log_debug "找到 x-cursor-checksum 设置代码"
|
|
echo "[FOUND] 找到 x-cursor-checksum 设置代码" >> "$LOG_FILE"
|
|
|
|
# 记录匹配的行到日志
|
|
grep -n 'i.header.set("x-cursor-checksum' "$file" >> "$LOG_FILE"
|
|
|
|
# 执行特定的替换
|
|
if sed -i.tmp 's/i\.header\.set("x-cursor-checksum",e===void 0?`${p}${t}`:`${p}${t}\/${e}`)/i.header.set("x-cursor-checksum",e===void 0?`${p}${t}`:`${p}${t}\/${p}`)/' "$file"; then
|
|
log_info "成功修改 x-cursor-checksum 设置代码"
|
|
echo "[SUCCESS] 成功完成 x-cursor-checksum 设置代码替换" >> "$LOG_FILE"
|
|
# 记录修改后的行
|
|
grep -n 'i.header.set("x-cursor-checksum' "$file" >> "$LOG_FILE"
|
|
((modified_count++))
|
|
log_info "成功修改文件: ${file/$temp_dir\//}"
|
|
else
|
|
log_error "修改 x-cursor-checksum 设置代码失败"
|
|
cp "${file}.bak" "$file"
|
|
fi
|
|
else
|
|
log_warn "未找到 x-cursor-checksum 设置代码"
|
|
echo "[FILE_CHECK] 未找到 x-cursor-checksum 设置代码" >> "$LOG_FILE"
|
|
|
|
# 记录文件部分内容到日志以便排查
|
|
echo "[FILE_CONTENT] 文件中包含 'header.set' 的行:" >> "$LOG_FILE"
|
|
grep -n "header.set" "$file" | head -20 >> "$LOG_FILE"
|
|
|
|
echo "[FILE_CONTENT] 文件中包含 'checksum' 的行:" >> "$LOG_FILE"
|
|
grep -n "checksum" "$file" | head -20 >> "$LOG_FILE"
|
|
fi
|
|
|
|
echo "[PROCESS_DETAIL] 完成处理 extensionHostProcess.js 文件" >> "$LOG_FILE"
|
|
elif grep -q "IOPlatformUUID" "$file"; then
|
|
log_debug "找到 IOPlatformUUID 关键字"
|
|
echo "[FOUND] 找到 IOPlatformUUID 关键字" >> "$LOG_FILE"
|
|
grep -n "IOPlatformUUID" "$file" | head -5 >> "$LOG_FILE"
|
|
|
|
# 定位 IOPlatformUUID 相关函数
|
|
if grep -q "function a\$" "$file"; then
|
|
# 检查是否已经修改过
|
|
if grep -q "return crypto.randomUUID()" "$file"; then
|
|
log_info "文件已经包含 randomUUID 调用,跳过修改"
|
|
((modified_count++))
|
|
continue
|
|
fi
|
|
|
|
# 针对 main.js 中发现的代码结构进行修改
|
|
if sed -i.tmp 's/function a\$(t){switch/function a\$(t){return crypto.randomUUID(); switch/' "$file"; then
|
|
log_debug "成功注入 randomUUID 调用到 a\$ 函数"
|
|
((modified_count++))
|
|
log_info "成功修改文件: ${file/$temp_dir\//}"
|
|
else
|
|
log_error "修改 a\$ 函数失败"
|
|
cp "${file}.bak" "$file"
|
|
fi
|
|
elif grep -q "async function v5" "$file"; then
|
|
# 检查是否已经修改过
|
|
if grep -q "return crypto.randomUUID()" "$file"; then
|
|
log_info "文件已经包含 randomUUID 调用,跳过修改"
|
|
((modified_count++))
|
|
continue
|
|
fi
|
|
|
|
# 替代方法 - 修改 v5 函数
|
|
if sed -i.tmp 's/async function v5(t){let e=/async function v5(t){return crypto.randomUUID(); let e=/' "$file"; then
|
|
log_debug "成功注入 randomUUID 调用到 v5 函数"
|
|
((modified_count++))
|
|
log_info "成功修改文件: ${file/$temp_dir\//}"
|
|
else
|
|
log_error "修改 v5 函数失败"
|
|
cp "${file}.bak" "$file"
|
|
fi
|
|
else
|
|
# 检查是否已经注入了自定义代码
|
|
if grep -q "// Cursor ID 修改工具注入" "$file"; then
|
|
log_info "文件已经包含自定义注入代码,跳过修改"
|
|
((modified_count++))
|
|
continue
|
|
fi
|
|
|
|
# 新增检查:检查是否已存在 randomDeviceId_ 时间戳模式
|
|
if grep -q "const randomDeviceId_[0-9]\\{10,\\}" "$file"; then
|
|
log_info "文件已经包含 randomDeviceId_ 模式,跳过通用注入"
|
|
echo "[FOUND] 文件已包含 randomDeviceId_ 模式,跳过通用注入: $file" >> "$LOG_FILE"
|
|
((modified_count++)) # 计为已修改,防止后续尝试其他方法
|
|
continue
|
|
fi
|
|
|
|
# 使用更通用的注入方法
|
|
log_warn "未找到具体函数,尝试使用通用修改方法"
|
|
inject_code="
|
|
// Cursor ID 修改工具注入 - $(date +%Y%m%d%H%M%S) - ES模块兼容版本
|
|
// 随机设备ID生成器注入 - $(date +%s)
|
|
import crypto from 'crypto';
|
|
|
|
const randomDeviceId_$(date +%s) = () => {
|
|
try {
|
|
return crypto.randomUUID();
|
|
} catch (e) {
|
|
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, c => {
|
|
const r = Math.random() * 16 | 0;
|
|
return (c === 'x' ? r : (r & 0x3 | 0x8)).toString(16);
|
|
});
|
|
}
|
|
};
|
|
"
|
|
# 将代码注入到文件开头
|
|
echo "$inject_code" > "${file}.new"
|
|
cat "$file" >> "${file}.new"
|
|
mv "${file}.new" "$file"
|
|
|
|
# 替换调用点
|
|
sed -i.tmp 's/await v5(!1)/randomDeviceId_'"$(date +%s)"'()/g' "$file"
|
|
sed -i.tmp 's/a\$(t)/randomDeviceId_'"$(date +%s)"'()/g' "$file"
|
|
|
|
log_debug "完成通用修改"
|
|
((modified_count++))
|
|
log_info "使用通用方法成功修改文件: ${file/$temp_dir\//}"
|
|
fi
|
|
else
|
|
# 未找到 IOPlatformUUID,可能是文件结构变化
|
|
log_warn "未找到 IOPlatformUUID,尝试替代方法"
|
|
|
|
# 检查是否已经注入或修改过
|
|
if grep -q "return crypto.randomUUID()" "$file" || grep -q "// Cursor ID 修改工具注入" "$file"; then
|
|
log_info "文件已经被修改过,跳过修改"
|
|
((modified_count++))
|
|
continue
|
|
fi
|
|
|
|
# 尝试找其他关键函数如 getMachineId 或 getDeviceId
|
|
if grep -q "function t\$()" "$file" || grep -q "async function y5" "$file"; then
|
|
log_debug "找到设备ID相关函数"
|
|
|
|
# 修改 MAC 地址获取函数
|
|
if grep -q "function t\$()" "$file"; then
|
|
sed -i.tmp 's/function t\$(){/function t\$(){return "00:00:00:00:00:00";/' "$file"
|
|
log_debug "修改 MAC 地址获取函数成功"
|
|
fi
|
|
|
|
# 修改设备ID获取函数
|
|
if grep -q "async function y5" "$file"; then
|
|
sed -i.tmp 's/async function y5(t){/async function y5(t){return crypto.randomUUID();/' "$file"
|
|
log_debug "修改设备ID获取函数成功"
|
|
fi
|
|
|
|
((modified_count++))
|
|
log_info "使用替代方法成功修改文件: ${file/$temp_dir\//}"
|
|
else
|
|
# 最后尝试的通用方法 - 在文件顶部插入重写函数定义
|
|
log_warn "未找到任何已知函数,使用最通用的方法"
|
|
|
|
inject_universal_code="
|
|
// Cursor ID 修改工具注入 - $(date +%Y%m%d%H%M%S) - ES模块兼容版本
|
|
// 全局拦截设备标识符 - $(date +%s)
|
|
import crypto from 'crypto';
|
|
|
|
// 保存原始函数引用
|
|
const originalRandomUUID_$(date +%s) = crypto.randomUUID;
|
|
|
|
// 重写crypto.randomUUID方法
|
|
crypto.randomUUID = function() {
|
|
return '${new_uuid}';
|
|
};
|
|
|
|
// 覆盖所有可能的系统ID获取函数 - 使用globalThis
|
|
globalThis.getMachineId = function() { return '${machine_id}'; };
|
|
globalThis.getDeviceId = function() { return '${device_id}'; };
|
|
globalThis.macMachineId = '${mac_machine_id}';
|
|
|
|
// 确保在不同环境下都能访问
|
|
if (typeof window !== 'undefined') {
|
|
window.getMachineId = globalThis.getMachineId;
|
|
window.getDeviceId = globalThis.getDeviceId;
|
|
window.macMachineId = globalThis.macMachineId;
|
|
}
|
|
|
|
// 确保模块顶层执行
|
|
console.log('Cursor全局设备标识符拦截已激活 - ES模块版本');
|
|
"
|
|
# 将代码注入到文件开头
|
|
local new_uuid=$(generate_uuid)
|
|
local machine_id="auth0|user_$(generate_hex_bytes 16)"
|
|
local device_id=$(generate_uuid)
|
|
local mac_machine_id=$(generate_hex_bytes 32)
|
|
|
|
inject_universal_code=${inject_universal_code//\$\{new_uuid\}/$new_uuid}
|
|
inject_universal_code=${inject_universal_code//\$\{machine_id\}/$machine_id}
|
|
inject_universal_code=${inject_universal_code//\$\{device_id\}/$device_id}
|
|
inject_universal_code=${inject_universal_code//\$\{mac_machine_id\}/$mac_machine_id}
|
|
|
|
echo "$inject_universal_code" > "${file}.new"
|
|
cat "$file" >> "${file}.new"
|
|
mv "${file}.new" "$file"
|
|
|
|
log_debug "完成通用覆盖"
|
|
((modified_count++))
|
|
log_info "使用最通用方法成功修改文件: ${file/$temp_dir\//}"
|
|
fi
|
|
fi
|
|
|
|
# 添加在关键操作后记录日志
|
|
echo "[MODIFIED] 文件修改后内容:" >> "$LOG_FILE"
|
|
grep -n "return crypto.randomUUID()" "$file" | head -3 >> "$LOG_FILE"
|
|
|
|
# 清理临时文件
|
|
rm -f "${file}.tmp" "${file}.bak"
|
|
echo "[PROCESS] 文件处理完成: $file" >> "$LOG_FILE"
|
|
done
|
|
|
|
if [ "$modified_count" -eq 0 ]; then
|
|
log_error "未能成功修改任何文件"
|
|
rm -rf "$temp_dir"
|
|
return 1
|
|
fi
|
|
|
|
# 重新签名应用(增加重试机制)
|
|
local max_retry=3
|
|
local retry_count=0
|
|
local sign_success=false
|
|
|
|
while [ $retry_count -lt $max_retry ]; do
|
|
((retry_count++))
|
|
log_info "尝试签名 (第 $retry_count 次)..."
|
|
|
|
# 使用更详细的签名参数
|
|
if codesign --sign - --force --deep --preserve-metadata=entitlements,identifier,flags "$temp_app" 2>&1 | tee /tmp/codesign.log; then
|
|
# 验证签名
|
|
if codesign --verify -vvvv "$temp_app" 2>/dev/null; then
|
|
sign_success=true
|
|
log_info "应用签名验证通过"
|
|
break
|
|
else
|
|
log_warn "签名验证失败,错误日志:"
|
|
cat /tmp/codesign.log
|
|
fi
|
|
else
|
|
log_warn "签名失败,错误日志:"
|
|
cat /tmp/codesign.log
|
|
fi
|
|
|
|
sleep 3
|
|
done
|
|
|
|
if ! $sign_success; then
|
|
log_error "经过 $max_retry 次尝试仍无法完成签名"
|
|
log_error "请手动执行以下命令完成签名:"
|
|
echo -e "${BLUE}sudo codesign --sign - --force --deep '${temp_app}'${NC}"
|
|
echo -e "${YELLOW}操作完成后,请手动将应用复制到原路径:${NC}"
|
|
echo -e "${BLUE}sudo cp -R '${temp_app}' '/Applications/'${NC}"
|
|
log_info "临时文件保留在:${temp_dir}"
|
|
return 1
|
|
fi
|
|
|
|
# 替换原应用
|
|
log_info "安装修改版应用..."
|
|
if ! sudo rm -rf "$CURSOR_APP_PATH" || ! sudo cp -R "$temp_app" "/Applications/"; then
|
|
log_error "应用替换失败,正在恢复..."
|
|
sudo rm -rf "$CURSOR_APP_PATH"
|
|
sudo cp -R "$backup_app" "$CURSOR_APP_PATH"
|
|
rm -rf "$temp_dir" "$backup_app"
|
|
return 1
|
|
fi
|
|
|
|
# 清理临时文件
|
|
rm -rf "$temp_dir" "$backup_app"
|
|
|
|
# 设置权限
|
|
sudo chown -R "$CURRENT_USER:staff" "$CURSOR_APP_PATH"
|
|
sudo chmod -R 755 "$CURSOR_APP_PATH"
|
|
|
|
log_info "Cursor 主程序文件修改完成!原版备份在: $backup_app"
|
|
return 0
|
|
}
|
|
|
|
# 显示文件树结构
|
|
show_file_tree() {
|
|
local base_dir=$(dirname "$STORAGE_FILE")
|
|
echo
|
|
log_info "文件结构:"
|
|
echo -e "${BLUE}$base_dir${NC}"
|
|
echo "├── globalStorage"
|
|
echo "│ ├── storage.json (已修改)"
|
|
echo "│ └── backups"
|
|
|
|
# 列出备份文件
|
|
if [ -d "$BACKUP_DIR" ]; then
|
|
local backup_files=("$BACKUP_DIR"/*)
|
|
if [ ${#backup_files[@]} -gt 0 ]; then
|
|
for file in "${backup_files[@]}"; do
|
|
if [ -f "$file" ]; then
|
|
echo "│ └── $(basename "$file")"
|
|
fi
|
|
done
|
|
else
|
|
echo "│ └── (空)"
|
|
fi
|
|
fi
|
|
echo
|
|
}
|
|
|
|
# 显示公众号信息
|
|
show_follow_info() {
|
|
echo
|
|
echo -e "${GREEN}================================${NC}"
|
|
echo -e "${YELLOW} 关注公众号【煎饼果子卷AI】一起交流更多Cursor技巧和AI知识(脚本免费、关注公众号加群有更多技巧和大佬) ${NC}"
|
|
echo -e "${GREEN}================================${NC}"
|
|
echo
|
|
}
|
|
|
|
# 禁用自动更新
|
|
disable_auto_update() {
|
|
local updater_path="$TARGET_HOME/Library/Application Support/Caches/cursor-updater"
|
|
local app_update_yml="/Applications/Cursor.app/Contents/Resources/app-update.yml"
|
|
|
|
echo
|
|
log_info "正在禁用 Cursor 自动更新..."
|
|
|
|
# 备份并清空 app-update.yml
|
|
if [ -f "$app_update_yml" ]; then
|
|
log_info "备份并修改 app-update.yml..."
|
|
if ! sudo cp "$app_update_yml" "${app_update_yml}.bak" 2>/dev/null; then
|
|
log_warn "备份 app-update.yml 失败,继续执行..."
|
|
fi
|
|
|
|
if sudo bash -c "echo '' > \"$app_update_yml\"" && \
|
|
sudo chmod 444 "$app_update_yml"; then
|
|
log_info "成功禁用 app-update.yml"
|
|
else
|
|
log_error "修改 app-update.yml 失败,请手动执行以下命令:"
|
|
echo -e "${BLUE}sudo cp \"$app_update_yml\" \"${app_update_yml}.bak\"${NC}"
|
|
echo -e "${BLUE}sudo bash -c 'echo \"\" > \"$app_update_yml\"'${NC}"
|
|
echo -e "${BLUE}sudo chmod 444 \"$app_update_yml\"${NC}"
|
|
fi
|
|
else
|
|
log_warn "未找到 app-update.yml 文件"
|
|
fi
|
|
|
|
# 同时也处理 cursor-updater
|
|
log_info "处理 cursor-updater..."
|
|
if sudo rm -rf "$updater_path" && \
|
|
sudo touch "$updater_path" && \
|
|
sudo chmod 444 "$updater_path"; then
|
|
log_info "成功禁用 cursor-updater"
|
|
else
|
|
log_error "禁用 cursor-updater 失败,请手动执行以下命令:"
|
|
echo -e "${BLUE}sudo rm -rf \"$updater_path\" && sudo touch \"$updater_path\" && sudo chmod 444 \"$updater_path\"${NC}"
|
|
fi
|
|
|
|
echo
|
|
log_info "验证方法:"
|
|
echo "1. 运行命令:ls -l \"$updater_path\""
|
|
echo " 确认文件权限显示为:r--r--r--"
|
|
echo "2. 运行命令:ls -l \"$app_update_yml\""
|
|
echo " 确认文件权限显示为:r--r--r--"
|
|
echo
|
|
log_info "完成后请重启 Cursor"
|
|
}
|
|
|
|
# 新增恢复功能选项
|
|
restore_feature() {
|
|
# 检查备份目录是否存在
|
|
if [ ! -d "$BACKUP_DIR" ]; then
|
|
log_warn "备份目录不存在"
|
|
return 1
|
|
fi
|
|
|
|
# 使用 find 命令获取备份文件列表并存储到数组
|
|
backup_files=()
|
|
while IFS= read -r file; do
|
|
[ -f "$file" ] && backup_files+=("$file")
|
|
done < <(find "$BACKUP_DIR" -name "*.backup_*" -type f 2>/dev/null | sort)
|
|
|
|
# 检查是否找到备份文件
|
|
if [ ${#backup_files[@]} -eq 0 ]; then
|
|
log_warn "未找到任何备份文件"
|
|
return 1
|
|
fi
|
|
|
|
echo
|
|
log_info "可用的备份文件:"
|
|
|
|
# 构建菜单选项字符串
|
|
menu_options="退出 - 不恢复任何文件"
|
|
for i in "${!backup_files[@]}"; do
|
|
menu_options="$menu_options|$(basename "${backup_files[$i]}")"
|
|
done
|
|
|
|
# 使用菜单选择函数
|
|
select_menu_option "请使用上下箭头选择要恢复的备份文件,按Enter确认:" "$menu_options" 0
|
|
choice=$?
|
|
|
|
# 处理用户输入
|
|
if [ "$choice" = "0" ]; then
|
|
log_info "跳过恢复操作"
|
|
return 0
|
|
fi
|
|
|
|
# 获取选择的备份文件 (减1是因为第一个选项是"退出")
|
|
local selected_backup="${backup_files[$((choice-1))]}"
|
|
|
|
# 验证文件存在性和可读性
|
|
if [ ! -f "$selected_backup" ] || [ ! -r "$selected_backup" ]; then
|
|
log_error "无法访问选择的备份文件"
|
|
return 1
|
|
fi
|
|
|
|
# 尝试恢复配置
|
|
if cp "$selected_backup" "$STORAGE_FILE"; then
|
|
chmod 644 "$STORAGE_FILE"
|
|
chown "$CURRENT_USER" "$STORAGE_FILE"
|
|
log_info "已从备份文件恢复配置: $(basename "$selected_backup")"
|
|
return 0
|
|
else
|
|
log_error "恢复配置失败"
|
|
return 1
|
|
fi
|
|
}
|
|
|
|
# 解决"应用已损坏,无法打开"问题
|
|
fix_damaged_app() {
|
|
log_info "正在修复"应用已损坏"问题..."
|
|
|
|
# 检查Cursor应用是否存在
|
|
if [ ! -d "$CURSOR_APP_PATH" ]; then
|
|
log_error "未找到Cursor应用: $CURSOR_APP_PATH"
|
|
return 1
|
|
fi
|
|
|
|
log_info "尝试移除隔离属性..."
|
|
if sudo find "$CURSOR_APP_PATH" -print0 \
|
|
| xargs -0 sudo xattr -d com.apple.quarantine 2>/dev/null
|
|
then
|
|
log_info "成功移除隔离属性"
|
|
else
|
|
log_warn "移除隔离属性失败,尝试其他方法..."
|
|
fi
|
|
|
|
log_info "尝试重新签名应用..."
|
|
if sudo codesign --force --deep --sign - "$CURSOR_APP_PATH" 2>/dev/null; then
|
|
log_info "应用重新签名成功"
|
|
else
|
|
log_warn "应用重新签名失败"
|
|
fi
|
|
|
|
echo
|
|
log_info "修复完成!请尝试重新打开Cursor应用"
|
|
echo
|
|
echo -e "${YELLOW}如果仍然无法打开,您可以尝试以下方法:${NC}"
|
|
echo "1. 在系统偏好设置->安全性与隐私中,点击"仍要打开"按钮"
|
|
echo "2. 暂时关闭Gatekeeper(不建议): sudo spctl --master-disable"
|
|
echo "3. 重新下载安装Cursor应用"
|
|
echo
|
|
echo -e "${BLUE} 参考链接: https://sysin.org/blog/macos-if-crashes-when-opening/ ${NC}"
|
|
|
|
return 0
|
|
}
|
|
|
|
# 新增:通用菜单选择函数
|
|
# 参数:
|
|
# $1 - 提示信息
|
|
# $2 - 选项数组,格式为 "选项1|选项2|选项3"
|
|
# $3 - 默认选项索引 (从0开始)
|
|
# 返回: 选中的选项索引 (从0开始)
|
|
select_menu_option() {
|
|
local prompt="$1"
|
|
IFS='|' read -ra options <<< "$2"
|
|
local default_index=${3:-0}
|
|
local selected_index=$default_index
|
|
local key_input
|
|
local cursor_up='\033[A'
|
|
local cursor_down='\033[B'
|
|
local enter_key=$'\n'
|
|
|
|
# 保存光标位置
|
|
tput sc
|
|
|
|
# 显示提示信息
|
|
echo -e "$prompt"
|
|
|
|
# 第一次显示菜单
|
|
for i in "${!options[@]}"; do
|
|
if [ $i -eq $selected_index ]; then
|
|
echo -e " ${GREEN}►${NC} ${options[$i]}"
|
|
else
|
|
echo -e " ${options[$i]}"
|
|
fi
|
|
done
|
|
|
|
# 循环处理键盘输入
|
|
while true; do
|
|
# 读取单个按键
|
|
read -rsn3 key_input
|
|
|
|
# 检测按键
|
|
case "$key_input" in
|
|
# 上箭头键
|
|
$'\033[A')
|
|
if [ $selected_index -gt 0 ]; then
|
|
((selected_index--))
|
|
fi
|
|
;;
|
|
# 下箭头键
|
|
$'\033[B')
|
|
if [ $selected_index -lt $((${#options[@]}-1)) ]; then
|
|
((selected_index++))
|
|
fi
|
|
;;
|
|
# Enter键
|
|
"")
|
|
echo # 换行
|
|
log_info "您选择了: ${options[$selected_index]}"
|
|
return $selected_index
|
|
;;
|
|
esac
|
|
|
|
# 恢复光标位置
|
|
tput rc
|
|
|
|
# 重新显示菜单
|
|
for i in "${!options[@]}"; do
|
|
if [ $i -eq $selected_index ]; then
|
|
echo -e " ${GREEN}►${NC} ${options[$i]}"
|
|
else
|
|
echo -e " ${options[$i]}"
|
|
fi
|
|
done
|
|
done
|
|
}
|
|
|
|
# 主函数
|
|
main() {
|
|
|
|
# 在显示菜单/流程说明前调整终端窗口大小;不支持则静默忽略
|
|
try_resize_terminal_window
|
|
|
|
# 初始化日志文件
|
|
initialize_log
|
|
log_info "脚本启动..."
|
|
|
|
# 🚀 启动时权限修复(解决EACCES错误)
|
|
log_info "🚀 [启动时权限] 执行启动时权限修复..."
|
|
ensure_cursor_directory_permissions
|
|
|
|
# 记录系统信息
|
|
log_info "系统信息: $(uname -a)"
|
|
log_info "当前用户: $CURRENT_USER"
|
|
log_cmd_output "sw_vers" "macOS 版本信息"
|
|
log_cmd_output "which codesign" "codesign 路径"
|
|
log_cmd_output "ls -ld "$CURSOR_APP_PATH"" "Cursor 应用信息"
|
|
|
|
# 新增环境检查
|
|
if [[ $(uname) != "Darwin" ]]; then
|
|
log_error "本脚本仅支持 macOS 系统"
|
|
exit 1
|
|
fi
|
|
|
|
clear
|
|
# 显示 Logo
|
|
echo -e "
|
|
██████╗██╗ ██╗██████╗ ███████╗ ██████╗ ██████╗
|
|
██╔════╝██║ ██║██╔══██╗██╔════╝██╔═══██╗██╔══██╗
|
|
██║ ██║ ██║██████╔╝███████╗██║ ██║██████╔╝
|
|
██║ ██║ ██║██╔══██╗╚════██║██║ ██║██╔══██╗
|
|
╚██████╗╚██████╔╝██║ ██║███████║╚██████╔╝██║ ██║
|
|
╚═════╝ ╚═════╝ ╚═╝ ╚═╝╚══════╝ ╚═════╝ ╚═╝ ╚═╝
|
|
"
|
|
echo -e "${BLUE}================================${NC}"
|
|
echo -e "${GREEN}🚀 Cursor 防掉试用Pro删除工具 ${NC}"
|
|
echo -e "${YELLOW}📱 关注公众号【煎饼果子卷AI】 ${NC}"
|
|
echo -e "${YELLOW}🤝 一起交流更多Cursor技巧和AI知识(脚本免费、关注公众号加群有更多技巧和大佬) ${NC}"
|
|
echo -e "${BLUE}================================${NC}"
|
|
echo
|
|
echo -e "${YELLOW}⚡ [小小广告] Cursor官网正规成品号:Pro¥65 | Pro+¥265 | Ultra¥888 独享账号| ¥488 Team绝版次数号1000次+20刀额度 | 全部7天质保 | ,WeChat:JavaRookie666 ${NC}"
|
|
echo
|
|
echo -e "${YELLOW}💡 [重要提示]${NC} 本工具采用分阶段执行策略,既能彻底清理又能修改机器码。"
|
|
echo -e "${YELLOW}💡 [重要提示]${NC} 本工具免费,如果对您有帮助,请关注公众号【煎饼果子卷AI】"
|
|
echo
|
|
echo
|
|
|
|
# 🎯 用户选择菜单
|
|
echo
|
|
echo -e "${GREEN}🎯 [选择模式]${NC} 请选择您要执行的操作:"
|
|
echo
|
|
echo -e "${BLUE} 1️⃣ 仅修改机器码${NC}"
|
|
echo -e "${YELLOW} • 仅执行机器码修改功能${NC}"
|
|
echo -e "${YELLOW} • 同步执行 JS 内核注入${NC}"
|
|
echo -e "${YELLOW} • 跳过文件夹删除/环境重置步骤${NC}"
|
|
echo -e "${YELLOW} • 保留现有Cursor配置和数据${NC}"
|
|
echo
|
|
echo -e "${BLUE} 2️⃣ 重置环境+修改机器码${NC}"
|
|
echo -e "${RED} • 执行完全环境重置(删除Cursor文件夹)${NC}"
|
|
echo -e "${RED} • ⚠️ 配置将丢失,请注意备份${NC}"
|
|
echo -e "${YELLOW} • 按照机器代码修改${NC}"
|
|
echo -e "${YELLOW} • 这相当于当前的完整脚本行为${NC}"
|
|
echo
|
|
|
|
# 获取用户选择
|
|
while true; do
|
|
read -p "请输入选择 (1 或 2): " user_choice
|
|
if [ "$user_choice" = "1" ]; then
|
|
echo -e "${GREEN}✅ [选择]${NC} 您选择了:仅修改机器码"
|
|
execute_mode="MODIFY_ONLY"
|
|
break
|
|
elif [ "$user_choice" = "2" ]; then
|
|
echo -e "${GREEN}✅ [选择]${NC} 您选择了:重置环境+修改机器码"
|
|
echo -e "${RED}⚠️ [重要警告]${NC} 此操作将删除所有Cursor配置文件!"
|
|
read -p "确认执行完全重置?(输入 yes 确认,其他任意键取消): " confirm_reset
|
|
if [ "$confirm_reset" = "yes" ]; then
|
|
execute_mode="RESET_AND_MODIFY"
|
|
break
|
|
else
|
|
echo -e "${YELLOW}👋 [取消]${NC} 用户取消重置操作"
|
|
continue
|
|
fi
|
|
else
|
|
echo -e "${RED}❌ [错误]${NC} 无效选择,请输入 1 或 2"
|
|
fi
|
|
done
|
|
|
|
echo
|
|
|
|
# 📋 根据选择显示执行流程说明
|
|
if [ "$execute_mode" = "MODIFY_ONLY" ]; then
|
|
echo -e "${GREEN}📋 [执行流程]${NC} 仅修改机器码模式将按以下步骤执行:"
|
|
echo -e "${BLUE} 1️⃣ 检测Cursor配置文件${NC}"
|
|
echo -e "${BLUE} 2️⃣ 备份现有配置文件${NC}"
|
|
echo -e "${BLUE} 3️⃣ 修改机器码配置${NC}"
|
|
echo -e "${BLUE} 4️⃣ 执行 JS 内核注入${NC}"
|
|
echo -e "${BLUE} 5️⃣ 显示操作完成信息${NC}"
|
|
echo
|
|
echo -e "${YELLOW}⚠️ [注意事项]${NC}"
|
|
echo -e "${YELLOW} • 不会删除任何文件夹或重置环境${NC}"
|
|
echo -e "${YELLOW} • 保留所有现有配置和数据${NC}"
|
|
echo -e "${YELLOW} • 原配置文件会自动备份${NC}"
|
|
echo -e "${YELLOW} • 需要Python3环境来处理JSON配置文件${NC}"
|
|
else
|
|
echo -e "${GREEN}📋 [执行流程]${NC} 重置环境+修改机器码模式将按以下步骤执行:"
|
|
echo -e "${BLUE} 1️⃣ 检测并关闭Cursor进程${NC}"
|
|
echo -e "${BLUE} 2️⃣ 保存Cursor程序路径信息${NC}"
|
|
echo -e "${BLUE} 3️⃣ 删除指定的Cursor试用相关文件夹${NC}"
|
|
echo -e "${BLUE} 📁 ~/Library/Application Support/Cursor${NC}"
|
|
echo -e "${BLUE} 📁 ~/.cursor${NC}"
|
|
echo -e "${BLUE} 3.5️⃣ 预创建必要目录结构,避免权限问题${NC}"
|
|
echo -e "${BLUE} 4️⃣ 重新启动Cursor让其生成新的配置文件${NC}"
|
|
echo -e "${BLUE} 5️⃣ 等待配置文件生成完成(最多45秒)${NC}"
|
|
echo -e "${BLUE} 6️⃣ 关闭Cursor进程${NC}"
|
|
echo -e "${BLUE} 7️⃣ 修改新生成的机器码配置文件${NC}"
|
|
echo -e "${BLUE} 8️⃣ 智能设备识别绕过(仅 JS 内核注入)${NC}"
|
|
echo -e "${BLUE} 9️⃣ 禁用自动更新${NC}"
|
|
echo -e "${BLUE} 🔟 显示操作完成统计信息${NC}"
|
|
echo
|
|
echo -e "${YELLOW}⚠️ [注意事项]${NC}"
|
|
echo -e "${YELLOW} • 脚本执行过程中请勿手动操作Cursor${NC}"
|
|
echo -e "${YELLOW} • 建议在执行前关闭所有Cursor窗口${NC}"
|
|
echo -e "${YELLOW} • 执行完成后需要重新启动Cursor${NC}"
|
|
echo -e "${YELLOW} • 原配置文件会自动备份到backups文件夹${NC}"
|
|
echo -e "${YELLOW} • 需要Python3环境来处理JSON配置文件${NC}"
|
|
echo -e "${YELLOW} • 已移除 MAC 地址修改,仅保留 JS 注入方案${NC}"
|
|
fi
|
|
echo
|
|
|
|
# 🤔 用户确认
|
|
echo -e "${GREEN}🤔 [确认]${NC} 请确认您已了解上述执行流程"
|
|
read -p "是否继续执行?(输入 y 或 yes 继续,其他任意键退出): " confirmation
|
|
if [[ ! "$confirmation" =~ ^(y|yes)$ ]]; then
|
|
echo -e "${YELLOW}👋 [退出]${NC} 用户取消执行,脚本退出"
|
|
exit 0
|
|
fi
|
|
echo -e "${GREEN}✅ [确认]${NC} 用户确认继续执行"
|
|
echo
|
|
|
|
# 🚀 根据用户选择执行相应功能
|
|
if [ "$execute_mode" = "MODIFY_ONLY" ]; then
|
|
log_info "🚀 [开始] 开始执行仅修改机器码功能..."
|
|
|
|
# 先进行环境检查
|
|
if ! test_cursor_environment "MODIFY_ONLY"; then
|
|
echo
|
|
log_error "❌ [环境检查失败] 无法继续执行"
|
|
echo
|
|
log_info "💡 [建议] 请选择以下操作:"
|
|
echo -e "${BLUE} 1️⃣ 选择'重置环境+修改机器码'选项(推荐)${NC}"
|
|
echo -e "${BLUE} 2️⃣ 手动启动Cursor一次,然后重新运行脚本${NC}"
|
|
echo -e "${BLUE} 3️⃣ 检查Cursor是否正确安装${NC}"
|
|
echo -e "${BLUE} 4️⃣ 安装Python3: brew install python3${NC}"
|
|
echo
|
|
read -p "按回车键退出..."
|
|
exit 1
|
|
fi
|
|
|
|
# 执行机器码修改
|
|
if modify_machine_code_config "MODIFY_ONLY"; then
|
|
echo
|
|
log_info "🎉 [完成] 机器码修改完成!"
|
|
log_info "💡 [提示] 现在可以启动Cursor使用新的机器码配置"
|
|
echo
|
|
log_info "🔧 [设备识别] 正在执行 JS 内核注入..."
|
|
if modify_cursor_js_files; then
|
|
log_info "✅ [设备识别] JS 内核注入完成"
|
|
else
|
|
log_warn "⚠️ [设备识别] JS 内核注入失败,请检查日志"
|
|
fi
|
|
else
|
|
echo
|
|
log_error "❌ [失败] 机器码修改失败!"
|
|
log_info "💡 [建议] 请尝试'重置环境+修改机器码'选项"
|
|
fi
|
|
# 🚫 禁用自动更新(仅修改模式也需要)
|
|
echo
|
|
log_info "🚫 [禁用更新] 正在禁用Cursor自动更新..."
|
|
disable_auto_update
|
|
|
|
# 🛡️ 关键修复:仅修改模式的权限修复
|
|
echo
|
|
log_info "🛡️ [权限修复] 执行仅修改模式的权限修复..."
|
|
log_info "💡 [说明] 确保Cursor应用能够正常启动,无权限错误"
|
|
ensure_cursor_directory_permissions
|
|
|
|
# 🔧 关键修复:修复应用签名问题(防止"应用已损坏"错误)
|
|
echo
|
|
log_info "🔧 [应用修复] 正在修复Cursor应用签名问题..."
|
|
log_info "💡 [说明] 防止出现'应用已损坏,无法打开'的错误"
|
|
|
|
if fix_damaged_app; then
|
|
log_info "✅ [应用修复] Cursor应用签名修复成功"
|
|
else
|
|
log_warn "⚠️ [应用修复] 应用签名修复失败,可能需要手动处理"
|
|
log_info "💡 [建议] 如果Cursor无法启动,请在系统偏好设置中允许打开"
|
|
fi
|
|
else
|
|
# 完整的重置环境+修改机器码流程
|
|
log_info "🚀 [开始] 开始执行重置环境+修改机器码功能..."
|
|
|
|
# 🚀 执行主要功能
|
|
check_permissions
|
|
check_and_kill_cursor
|
|
|
|
# 🚨 重要警告提示
|
|
echo
|
|
echo -e "${RED}🚨 [重要警告]${NC} ============================================"
|
|
log_warn "⚠️ [风控提醒] Cursor 风控机制非常严格!"
|
|
log_warn "⚠️ [必须删除] 必须完全删除指定文件夹,不能有任何残留设置"
|
|
log_warn "⚠️ [防掉试用] 只有彻底清理才能有效防止掉试用Pro状态"
|
|
echo -e "${RED}🚨 [重要警告]${NC} ============================================"
|
|
echo
|
|
|
|
# 🎯 执行 Cursor 防掉试用Pro删除文件夹功能
|
|
log_info "🚀 [开始] 开始执行核心功能..."
|
|
remove_cursor_trial_folders
|
|
|
|
# 🔄 重启Cursor让其重新生成配置文件
|
|
restart_cursor_and_wait
|
|
|
|
# 🛠️ 修改机器码配置
|
|
modify_machine_code_config
|
|
|
|
# 🔧 智能设备识别绕过(仅 JS 内核注入)
|
|
echo
|
|
log_info "🔧 [设备识别] 开始智能设备识别绕过..."
|
|
log_info "💡 [说明] 已移除 MAC 地址修改,直接使用 JS 内核注入"
|
|
if ! run_device_bypass; then
|
|
log_warn "⚠️ [设备识别] 智能设备识别绕过未完全成功,请查看日志"
|
|
fi
|
|
|
|
|
|
# 🔧 关键修复:修复应用签名问题(防止"应用已损坏"错误)
|
|
echo
|
|
log_info "🔧 [应用修复] 正在修复Cursor应用签名问题..."
|
|
log_info "💡 [说明] 防止出现'应用已损坏,无法打开'的错误"
|
|
|
|
if fix_damaged_app; then
|
|
log_info "✅ [应用修复] Cursor应用签名修复成功"
|
|
else
|
|
log_warn "⚠️ [应用修复] 应用签名修复失败,可能需要手动处理"
|
|
log_info "💡 [建议] 如果Cursor无法启动,请在系统偏好设置中允许打开"
|
|
fi
|
|
fi
|
|
|
|
# 🚫 禁用自动更新
|
|
echo
|
|
log_info "🚫 [禁用更新] 正在禁用Cursor自动更新..."
|
|
disable_auto_update
|
|
|
|
# 🎉 显示操作完成信息
|
|
echo
|
|
log_info "🎉 [完成] Cursor 防掉试用Pro删除操作已完成!"
|
|
echo
|
|
|
|
# 📱 显示公众号信息
|
|
echo -e "${GREEN}================================${NC}"
|
|
echo -e "${YELLOW}📱 关注公众号【煎饼果子卷AI】一起交流更多Cursor技巧和AI知识(脚本免费、关注公众号加群有更多技巧和大佬) ${NC}"
|
|
echo -e "${YELLOW}⚡ [小小广告] Cursor官网正规成品号:Pro¥65 | Pro+¥265 | Ultra¥888 独享账号| ¥488 Team绝版次数号1000次+20刀额度 | 全部7天质保 | ,WeChat:JavaRookie666 ${NC}"
|
|
echo -e "${GREEN}================================${NC}"
|
|
echo
|
|
log_info "🚀 [提示] 现在可以重新启动 Cursor 尝试使用了!"
|
|
echo
|
|
|
|
# 🎉 显示修改结果总结
|
|
echo
|
|
echo -e "${GREEN}================================${NC}"
|
|
echo -e "${BLUE} 🎯 修改结果总结 ${NC}"
|
|
echo -e "${GREEN}================================${NC}"
|
|
echo -e "${GREEN}✅ JSON配置文件修改: 完成${NC}"
|
|
echo -e "${GREEN}✅ 自动更新禁用: 完成${NC}"
|
|
echo -e "${GREEN}================================${NC}"
|
|
echo
|
|
|
|
# 🛡️ 脚本完成前最终权限修复
|
|
echo
|
|
log_info "🛡️ [最终权限修复] 执行脚本完成前的最终权限修复..."
|
|
ensure_cursor_directory_permissions
|
|
protect_storage_file
|
|
|
|
# 🎉 脚本执行完成
|
|
log_info "🎉 [完成] 所有操作已完成!"
|
|
echo
|
|
log_info "💡 [重要提示] 完整的Cursor破解流程已执行:"
|
|
echo -e "${BLUE} ✅ 机器码配置文件修改${NC}"
|
|
echo -e "${BLUE} ✅ 自动更新功能禁用${NC}"
|
|
echo -e "${BLUE} ✅ 权限修复和验证${NC}"
|
|
echo
|
|
log_warn "⚠️ [注意] 重启 Cursor 后生效"
|
|
echo
|
|
log_info "🚀 [下一步] 现在可以启动 Cursor 尝试使用了!"
|
|
echo
|
|
|
|
# 记录脚本完成信息
|
|
log_info "📝 [日志] 脚本执行完成"
|
|
echo "========== Cursor 防掉试用Pro删除工具日志结束 $(date) ==========" >> "$LOG_FILE"
|
|
|
|
# 显示日志文件位置
|
|
echo
|
|
log_info "📄 [日志] 详细日志已保存到: $LOG_FILE"
|
|
echo "如遇问题请将此日志文件提供给开发者以协助排查"
|
|
echo
|
|
}
|
|
|
|
# 执行主函数
|
|
main
|