mirror of
https://github.com/yeongpin/cursor-free-vip.git
synced 2025-04-24 08:25:23 +00:00
160 lines
7.0 KiB
Python
160 lines
7.0 KiB
Python
import os
|
||
import json
|
||
import shutil
|
||
import platform
|
||
import configparser
|
||
import time
|
||
from colorama import Fore, Style, init
|
||
import sys
|
||
import traceback
|
||
from utils import get_user_documents_path
|
||
|
||
# Initialize colorama
|
||
init()
|
||
|
||
# Define emoji constants
|
||
EMOJI = {
|
||
'INFO': 'ℹ️',
|
||
'SUCCESS': '✅',
|
||
'ERROR': '❌',
|
||
'WARNING': '⚠️',
|
||
'FILE': '📄',
|
||
'BACKUP': '💾',
|
||
'RESET': '🔄',
|
||
'VERSION': '🏷️'
|
||
}
|
||
|
||
def get_product_json_path(translator=None):
|
||
"""Get Cursor product.json path"""
|
||
system = platform.system()
|
||
|
||
# Read configuration
|
||
config_dir = os.path.join(get_user_documents_path(), ".cursor-free-vip")
|
||
config_file = os.path.join(config_dir, "config.ini")
|
||
config = configparser.ConfigParser()
|
||
|
||
if os.path.exists(config_file):
|
||
config.read(config_file)
|
||
|
||
if system == "Windows":
|
||
localappdata = os.environ.get("LOCALAPPDATA")
|
||
if not localappdata:
|
||
raise OSError(translator.get('bypass.localappdata_not_found') if translator else "LOCALAPPDATA environment variable not found")
|
||
|
||
product_json_path = os.path.join(localappdata, "Programs", "Cursor", "resources", "app", "product.json")
|
||
|
||
# Check if path exists in config
|
||
if 'WindowsPaths' in config and 'cursor_path' in config['WindowsPaths']:
|
||
cursor_path = config.get('WindowsPaths', 'cursor_path')
|
||
product_json_path = os.path.join(cursor_path, "product.json")
|
||
|
||
elif system == "Darwin": # macOS
|
||
product_json_path = "/Applications/Cursor.app/Contents/Resources/app/product.json"
|
||
if config.has_section('MacPaths') and config.has_option('MacPaths', 'product_json_path'):
|
||
product_json_path = config.get('MacPaths', 'product_json_path')
|
||
|
||
elif system == "Linux":
|
||
# Try multiple common paths
|
||
possible_paths = [
|
||
"/opt/Cursor/resources/app/product.json",
|
||
"/usr/share/cursor/resources/app/product.json",
|
||
"/usr/lib/cursor/app/product.json"
|
||
]
|
||
|
||
# Add extracted AppImage paths
|
||
extracted_usr_paths = os.path.expanduser("~/squashfs-root/usr/share/cursor/resources/app/product.json")
|
||
if os.path.exists(extracted_usr_paths):
|
||
possible_paths.append(extracted_usr_paths)
|
||
|
||
for path in possible_paths:
|
||
if os.path.exists(path):
|
||
product_json_path = path
|
||
break
|
||
else:
|
||
raise OSError(translator.get('bypass.product_json_not_found') if translator else "product.json not found in common Linux paths")
|
||
|
||
else:
|
||
raise OSError(translator.get('bypass.unsupported_os', system=system) if translator else f"Unsupported operating system: {system}")
|
||
|
||
if not os.path.exists(product_json_path):
|
||
raise OSError(translator.get('bypass.file_not_found', path=product_json_path) if translator else f"File not found: {product_json_path}")
|
||
|
||
return product_json_path
|
||
|
||
def compare_versions(version1, version2):
|
||
"""Compare two version strings"""
|
||
v1_parts = [int(x) for x in version1.split('.')]
|
||
v2_parts = [int(x) for x in version2.split('.')]
|
||
|
||
for i in range(max(len(v1_parts), len(v2_parts))):
|
||
v1 = v1_parts[i] if i < len(v1_parts) else 0
|
||
v2 = v2_parts[i] if i < len(v2_parts) else 0
|
||
if v1 < v2:
|
||
return -1
|
||
elif v1 > v2:
|
||
return 1
|
||
|
||
return 0
|
||
|
||
def bypass_version(translator=None):
|
||
"""Bypass Cursor version check by modifying product.json"""
|
||
try:
|
||
print(f"\n{Fore.CYAN}{EMOJI['INFO']} {translator.get('bypass.starting') if translator else 'Starting Cursor version bypass...'}{Style.RESET_ALL}")
|
||
|
||
# Get product.json path
|
||
product_json_path = get_product_json_path(translator)
|
||
print(f"{Fore.CYAN}{EMOJI['FILE']} {translator.get('bypass.found_product_json', path=product_json_path) if translator else f'Found product.json: {product_json_path}'}{Style.RESET_ALL}")
|
||
|
||
# Check file permissions
|
||
if not os.access(product_json_path, os.W_OK):
|
||
print(f"{Fore.RED}{EMOJI['ERROR']} {translator.get('bypass.no_write_permission', path=product_json_path) if translator else f'No write permission for file: {product_json_path}'}{Style.RESET_ALL}")
|
||
return False
|
||
|
||
# Read product.json
|
||
try:
|
||
with open(product_json_path, "r", encoding="utf-8") as f:
|
||
product_data = json.load(f)
|
||
except Exception as e:
|
||
print(f"{Fore.RED}{EMOJI['ERROR']} {translator.get('bypass.read_failed', error=str(e)) if translator else f'Failed to read product.json: {str(e)}'}{Style.RESET_ALL}")
|
||
return False
|
||
|
||
# Get current version
|
||
current_version = product_data.get("version", "0.0.0")
|
||
print(f"{Fore.CYAN}{EMOJI['VERSION']} {translator.get('bypass.current_version', version=current_version) if translator else f'Current version: {current_version}'}{Style.RESET_ALL}")
|
||
|
||
# Check if version needs to be modified
|
||
if compare_versions(current_version, "0.46.0") < 0:
|
||
# Create backup
|
||
timestamp = time.strftime("%Y%m%d%H%M%S")
|
||
backup_path = f"{product_json_path}.{timestamp}"
|
||
shutil.copy2(product_json_path, backup_path)
|
||
print(f"{Fore.GREEN}{EMOJI['BACKUP']} {translator.get('bypass.backup_created', path=backup_path) if translator else f'Backup created: {backup_path}'}{Style.RESET_ALL}")
|
||
|
||
# Modify version
|
||
new_version = "0.48.7"
|
||
product_data["version"] = new_version
|
||
|
||
# Save modified product.json
|
||
try:
|
||
with open(product_json_path, "w", encoding="utf-8") as f:
|
||
json.dump(product_data, f, indent=2)
|
||
print(f"{Fore.GREEN}{EMOJI['SUCCESS']} {translator.get('bypass.version_updated', old=current_version, new=new_version) if translator else f'Version updated from {current_version} to {new_version}'}{Style.RESET_ALL}")
|
||
return True
|
||
except Exception as e:
|
||
print(f"{Fore.RED}{EMOJI['ERROR']} {translator.get('bypass.write_failed', error=str(e)) if translator else f'Failed to write product.json: {str(e)}'}{Style.RESET_ALL}")
|
||
return False
|
||
else:
|
||
print(f"{Fore.YELLOW}{EMOJI['INFO']} {translator.get('bypass.no_update_needed', version=current_version) if translator else f'No update needed. Current version {current_version} is already >= 0.46.0'}{Style.RESET_ALL}")
|
||
return True
|
||
|
||
except Exception as e:
|
||
print(f"{Fore.RED}{EMOJI['ERROR']} {translator.get('bypass.bypass_failed', error=str(e)) if translator else f'Version bypass failed: {str(e)}'}{Style.RESET_ALL}")
|
||
print(f"{Fore.YELLOW}{EMOJI['INFO']} {translator.get('bypass.stack_trace') if translator else 'Stack trace'}: {traceback.format_exc()}{Style.RESET_ALL}")
|
||
return False
|
||
|
||
def main(translator=None):
|
||
"""Main function"""
|
||
return bypass_version(translator)
|
||
|
||
if __name__ == "__main__":
|
||
main() |