mirror of
https://github.com/yeongpin/cursor-free-vip.git
synced 2025-08-02 20:47:35 +08:00
238 lines
13 KiB
Python
238 lines
13 KiB
Python
import os
|
||
import sys
|
||
import configparser
|
||
from colorama import Fore, Style
|
||
from utils import get_user_documents_path, get_default_chrome_path, get_linux_cursor_path
|
||
|
||
EMOJI = {
|
||
"INFO": "ℹ️",
|
||
"WARNING": "⚠️",
|
||
"ERROR": "❌",
|
||
"SUCCESS": "✅",
|
||
"ADMIN": "🔒",
|
||
"ARROW": "➡️",
|
||
"USER": "👤",
|
||
"KEY": "🔑",
|
||
"SETTINGS": "⚙️"
|
||
}
|
||
|
||
def setup_config(translator=None):
|
||
"""Setup configuration file and return config object"""
|
||
try:
|
||
config_dir = os.path.join(get_user_documents_path(), ".cursor-free-vip")
|
||
config_file = os.path.join(config_dir, "config.ini")
|
||
os.makedirs(config_dir, exist_ok=True)
|
||
|
||
config = configparser.ConfigParser()
|
||
|
||
# Default configuration
|
||
default_config = {
|
||
'Chrome': {
|
||
'chromepath': get_default_chrome_path()
|
||
},
|
||
'Turnstile': {
|
||
'handle_turnstile_time': '2',
|
||
'handle_turnstile_random_time': '1-3'
|
||
},
|
||
'Timing': {
|
||
'min_random_time': '0.1',
|
||
'max_random_time': '0.8',
|
||
'page_load_wait': '0.1-0.8',
|
||
'input_wait': '0.3-0.8',
|
||
'submit_wait': '0.5-1.5',
|
||
'verification_code_input': '0.1-0.3',
|
||
'verification_success_wait': '2-3',
|
||
'verification_retry_wait': '2-3',
|
||
'email_check_initial_wait': '4-6',
|
||
'email_refresh_wait': '2-4',
|
||
'settings_page_load_wait': '1-2',
|
||
'failed_retry_time': '0.5-1',
|
||
'retry_interval': '8-12',
|
||
'max_timeout': '160'
|
||
},
|
||
'Utils': {
|
||
'enabled_update_check': 'True',
|
||
'enabled_account_info': 'True'
|
||
}
|
||
}
|
||
|
||
# Add system-specific path configuration
|
||
if sys.platform == "win32":
|
||
appdata = os.getenv("APPDATA")
|
||
localappdata = os.getenv("LOCALAPPDATA", "")
|
||
default_config['WindowsPaths'] = {
|
||
'storage_path': os.path.join(appdata, "Cursor", "User", "globalStorage", "storage.json"),
|
||
'sqlite_path': os.path.join(appdata, "Cursor", "User", "globalStorage", "state.vscdb"),
|
||
'machine_id_path': os.path.join(appdata, "Cursor", "machineId"),
|
||
'cursor_path': os.path.join(localappdata, "Programs", "Cursor", "resources", "app"),
|
||
'updater_path': os.path.join(localappdata, "cursor-updater"),
|
||
'update_yml_path': os.path.join(localappdata, "Programs", "Cursor", "resources", "app-update.yml")
|
||
}
|
||
# Create storage directory
|
||
os.makedirs(os.path.dirname(default_config['WindowsPaths']['storage_path']), exist_ok=True)
|
||
|
||
elif sys.platform == "darwin":
|
||
default_config['MacPaths'] = {
|
||
'storage_path': os.path.abspath(os.path.expanduser("~/Library/Application Support/Cursor/User/globalStorage/storage.json")),
|
||
'sqlite_path': os.path.abspath(os.path.expanduser("~/Library/Application Support/Cursor/User/globalStorage/state.vscdb")),
|
||
'machine_id_path': os.path.expanduser("~/Library/Application Support/Cursor/machineId"),
|
||
'cursor_path': "/Applications/Cursor.app/Contents/Resources/app",
|
||
'updater_path': os.path.expanduser("~/Library/Application Support/cursor-updater"),
|
||
'update_yml_path': "/Applications/Cursor.app/Contents/Resources/app-update.yml"
|
||
}
|
||
# Create storage directory
|
||
os.makedirs(os.path.dirname(default_config['MacPaths']['storage_path']), exist_ok=True)
|
||
|
||
elif sys.platform == "linux":
|
||
# Get the actual user's home directory, handling both sudo and normal cases
|
||
current_user = os.getenv('USER') or os.getenv('USERNAME') or os.getenv('SUDO_USER')
|
||
if not current_user:
|
||
current_user = os.path.expanduser('~').split('/')[-1]
|
||
|
||
actual_home = f"/home/{current_user}"
|
||
if not os.path.exists(actual_home):
|
||
actual_home = os.path.expanduser("~")
|
||
|
||
# Define base config directory
|
||
config_base = os.path.join(actual_home, ".config")
|
||
|
||
# Try both "Cursor" and "cursor" directory names
|
||
cursor_dir = None
|
||
for dir_name in ["Cursor", "cursor"]:
|
||
test_dir = os.path.join(config_base, dir_name)
|
||
if os.path.exists(test_dir):
|
||
cursor_dir = test_dir
|
||
break
|
||
|
||
if not cursor_dir:
|
||
print(f"{Fore.YELLOW}{EMOJI['WARNING']} {translator.get('config.neither_cursor_nor_cursor_directory_found', config_base=config_base)}{Style.RESET_ALL}")
|
||
print(f"{Fore.YELLOW}{EMOJI['INFO']} {translator.get('config.please_make_sure_cursor_is_installed_and_has_been_run_at_least_once')}{Style.RESET_ALL}")
|
||
|
||
# Define Linux paths using the found cursor directory
|
||
storage_path = os.path.abspath(os.path.join(cursor_dir, "User/globalStorage/storage.json")) if cursor_dir else ""
|
||
storage_dir = os.path.dirname(storage_path) if storage_path else ""
|
||
|
||
# Verify paths and permissions
|
||
try:
|
||
# Check storage directory
|
||
if storage_dir and not os.path.exists(storage_dir):
|
||
print(f"{Fore.YELLOW}{EMOJI['WARNING']} {translator.get('config.storage_directory_not_found', storage_dir=storage_dir)}{Style.RESET_ALL}")
|
||
print(f"{Fore.YELLOW}{EMOJI['INFO']} {translator.get('config.please_make_sure_cursor_is_installed_and_has_been_run_at_least_once')}{Style.RESET_ALL}")
|
||
|
||
# Check storage.json with more detailed verification
|
||
if storage_path and os.path.exists(storage_path):
|
||
# Get file stats
|
||
try:
|
||
stat = os.stat(storage_path)
|
||
print(f"{Fore.GREEN}{EMOJI['INFO']} {translator.get('config.storage_file_found', storage_path=storage_path)}{Style.RESET_ALL}")
|
||
print(f"{Fore.GREEN}{EMOJI['INFO']} {translator.get('config.file_size', stat.st_size)}{Style.RESET_ALL}")
|
||
print(f"{Fore.GREEN}{EMOJI['INFO']} {translator.get('config.file_permissions', oct(stat.st_mode & 0o777))}{Style.RESET_ALL}")
|
||
print(f"{Fore.GREEN}{EMOJI['INFO']} {translator.get('config.file_owner', stat.st_uid)}{Style.RESET_ALL}")
|
||
print(f"{Fore.GREEN}{EMOJI['INFO']} {translator.get('config.file_group', stat.st_gid)}{Style.RESET_ALL}")
|
||
except Exception as e:
|
||
print(f"{Fore.RED}{EMOJI['ERROR']} {translator.get('config.error_getting_file_stats', error=str(e))}{Style.RESET_ALL}")
|
||
|
||
# Check if file is readable and writable
|
||
if not os.access(storage_path, os.R_OK | os.W_OK):
|
||
print(f"{Fore.RED}{EMOJI['ERROR']} {translator.get('config.permission_denied', storage_path=storage_path)}{Style.RESET_ALL}")
|
||
print(f"{Fore.YELLOW}{EMOJI['INFO']} {translator.get('config.try_running', current_user=current_user, storage_path=storage_path)}{Style.RESET_ALL}")
|
||
print(f"{Fore.YELLOW}{EMOJI['INFO']} {translator.get('config.and', storage_path=storage_path)}{Style.RESET_ALL}")
|
||
|
||
# Try to read the file to verify it's not corrupted
|
||
try:
|
||
with open(storage_path, 'r') as f:
|
||
content = f.read()
|
||
if not content.strip():
|
||
print(f"{Fore.YELLOW}{EMOJI['WARNING']} {translator.get('config.storage_file_is_empty', storage_path=storage_path)}{Style.RESET_ALL}")
|
||
print(f"{Fore.YELLOW}{EMOJI['INFO']} {translator.get('config.please_make_sure_cursor_is_installed_and_has_been_run_at_least_once')}{Style.RESET_ALL}")
|
||
else:
|
||
print(f"{Fore.GREEN}{EMOJI['SUCCESS']} {translator.get('config.storage_file_is_valid_and_contains_data')}{Style.RESET_ALL}")
|
||
except Exception as e:
|
||
print(f"{Fore.RED}{EMOJI['ERROR']} {translator.get('config.error_reading_storage_file', error=str(e))}{Style.RESET_ALL}")
|
||
print(f"{Fore.YELLOW}{EMOJI['INFO']} {translator.get('config.the_file_might_be_corrupted_please_reinstall_cursor')}{Style.RESET_ALL}")
|
||
elif storage_path:
|
||
print(f"{Fore.YELLOW}{EMOJI['WARNING']} {translator.get('config.storage_file_not_found', storage_path=storage_path)}{Style.RESET_ALL}")
|
||
print(f"{Fore.YELLOW}{EMOJI['INFO']} {translator.get('config.please_make_sure_cursor_is_installed_and_has_been_run_at_least_once')}{Style.RESET_ALL}")
|
||
|
||
except (OSError, IOError) as e:
|
||
print(f"{Fore.RED}{EMOJI['ERROR']} {translator.get('config.error_checking_linux_paths', error=str(e))}{Style.RESET_ALL}")
|
||
|
||
# Define all paths using the found cursor directory
|
||
default_config['LinuxPaths'] = {
|
||
'storage_path': storage_path,
|
||
'sqlite_path': os.path.abspath(os.path.join(cursor_dir, "User/globalStorage/state.vscdb")) if cursor_dir else "",
|
||
'machine_id_path': os.path.join(cursor_dir, "machineid") if cursor_dir else "",
|
||
'cursor_path': get_linux_cursor_path(),
|
||
'updater_path': os.path.join(config_base, "cursor-updater"),
|
||
'update_yml_path': os.path.join(cursor_dir, "resources/app-update.yml") if cursor_dir else ""
|
||
}
|
||
|
||
# Read existing configuration and merge
|
||
if os.path.exists(config_file):
|
||
config.read(config_file, encoding='utf-8')
|
||
config_modified = False
|
||
|
||
for section, options in default_config.items():
|
||
if not config.has_section(section):
|
||
config.add_section(section)
|
||
config_modified = True
|
||
for option, value in options.items():
|
||
if not config.has_option(section, option):
|
||
config.set(section, option, str(value))
|
||
config_modified = True
|
||
if translator:
|
||
print(f"{Fore.YELLOW}{EMOJI['INFO']} {translator.get('register.config_option_added', option=f'{section}.{option}')}{Style.RESET_ALL}")
|
||
|
||
if config_modified:
|
||
with open(config_file, 'w', encoding='utf-8') as f:
|
||
config.write(f)
|
||
if translator:
|
||
print(f"{Fore.GREEN}{EMOJI['SUCCESS']} {translator.get('register.config_updated')}{Style.RESET_ALL}")
|
||
else:
|
||
for section, options in default_config.items():
|
||
config.add_section(section)
|
||
for option, value in options.items():
|
||
config.set(section, option, str(value))
|
||
|
||
with open(config_file, 'w', encoding='utf-8') as f:
|
||
config.write(f)
|
||
if translator:
|
||
print(f"{Fore.GREEN}{EMOJI['SUCCESS']} {translator.get('register.config_created')}: {config_file}{Style.RESET_ALL}")
|
||
|
||
return config
|
||
|
||
except Exception as e:
|
||
if translator:
|
||
print(f"{Fore.RED}{EMOJI['ERROR']} {translator.get('register.config_setup_error', error=str(e))}{Style.RESET_ALL}")
|
||
return None
|
||
|
||
def print_config(config, translator=None):
|
||
"""Print configuration in a readable format"""
|
||
if not config:
|
||
print(f"{Fore.YELLOW}{EMOJI['WARNING']} {translator.get('config.config_not_available')}{Style.RESET_ALL}")
|
||
return
|
||
|
||
print(f"\n{Fore.CYAN}{EMOJI['INFO']} {translator.get('config.configuration')}:{Style.RESET_ALL}")
|
||
print(f"\n{Fore.CYAN}{'─' * 70}{Style.RESET_ALL}")
|
||
for section in config.sections():
|
||
print(f"{Fore.GREEN}[{section}]{Style.RESET_ALL}")
|
||
for key, value in config.items(section):
|
||
# 对布尔值进行特殊处理,使其显示为彩色
|
||
if value.lower() in ('true', 'yes', 'on', '1'):
|
||
value_display = f"{Fore.GREEN}{translator.get('config.enabled')}{Style.RESET_ALL}"
|
||
elif value.lower() in ('false', 'no', 'off', '0'):
|
||
value_display = f"{Fore.RED}{translator.get('config.disabled')}{Style.RESET_ALL}"
|
||
else:
|
||
value_display = value
|
||
|
||
print(f" {key} = {value_display}")
|
||
|
||
print(f"\n{Fore.CYAN}{'─' * 70}{Style.RESET_ALL}")
|
||
config_dir = os.path.join(get_user_documents_path(), ".cursor-free-vip", "config.ini")
|
||
print(f"{Fore.CYAN}{EMOJI['INFO']} {translator.get('config.config_directory')}: {config_dir}{Style.RESET_ALL}")
|
||
|
||
print()
|
||
|
||
def get_config(translator=None):
|
||
"""Get existing config or create new one"""
|
||
return setup_config(translator) |