2024-12-28 20:05:29 +08:00
|
|
|
|
import os
|
2024-12-28 20:34:34 +08:00
|
|
|
|
|
|
|
|
|
|
os.environ["PYTHONVERBOSE"] = "0"
|
|
|
|
|
|
os.environ["PYINSTALLER_VERBOSE"] = "0"
|
2024-12-28 20:05:29 +08:00
|
|
|
|
|
2024-12-28 15:14:40 +08:00
|
|
|
|
from DrissionPage import ChromiumOptions, Chromium
|
|
|
|
|
|
from DrissionPage.common import Keys
|
|
|
|
|
|
import re
|
|
|
|
|
|
import time
|
|
|
|
|
|
import random
|
|
|
|
|
|
from cursor_auth_manager import CursorAuthManager
|
|
|
|
|
|
from configparser import ConfigParser
|
|
|
|
|
|
import os
|
|
|
|
|
|
import sys
|
2024-12-28 20:05:29 +08:00
|
|
|
|
import logging
|
2024-12-28 21:56:48 +08:00
|
|
|
|
from browser_utils import BrowserManager
|
|
|
|
|
|
from get_veri_code import EmailVerificationHandler
|
2024-12-28 20:05:29 +08:00
|
|
|
|
|
|
|
|
|
|
# 在文件开头设置日志
|
|
|
|
|
|
logging.basicConfig(
|
|
|
|
|
|
level=logging.WARNING,
|
2024-12-28 20:34:34 +08:00
|
|
|
|
format="%(asctime)s - %(levelname)s - %(message)s",
|
2024-12-28 20:05:29 +08:00
|
|
|
|
handlers=[
|
|
|
|
|
|
logging.StreamHandler(),
|
2024-12-28 20:34:34 +08:00
|
|
|
|
logging.FileHandler("cursor_keep_alive.log", encoding="utf-8"),
|
|
|
|
|
|
],
|
2024-12-28 20:05:29 +08:00
|
|
|
|
)
|
2024-12-28 15:14:40 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def load_config():
|
|
|
|
|
|
"""加载配置文件"""
|
|
|
|
|
|
config = ConfigParser()
|
|
|
|
|
|
|
2024-12-28 20:34:34 +08:00
|
|
|
|
# 获取程序运行的实际目录
|
|
|
|
|
|
if getattr(sys, "frozen", False):
|
|
|
|
|
|
# 打包后的情况:使用可执行文件所在目录
|
|
|
|
|
|
root_dir = os.path.dirname(sys.executable)
|
2024-12-28 20:05:29 +08:00
|
|
|
|
else:
|
2024-12-28 20:34:34 +08:00
|
|
|
|
# 开发环境:使用当前工作目录
|
2024-12-28 20:05:29 +08:00
|
|
|
|
root_dir = os.getcwd()
|
2024-12-28 20:34:34 +08:00
|
|
|
|
|
2024-12-28 15:14:40 +08:00
|
|
|
|
config_path = os.path.join(root_dir, "config.ini")
|
|
|
|
|
|
|
|
|
|
|
|
if os.path.exists(config_path):
|
|
|
|
|
|
config.read(config_path, encoding="utf-8")
|
2024-12-28 20:05:29 +08:00
|
|
|
|
print(f"已加载配置文件: {config_path}")
|
2024-12-28 20:34:34 +08:00
|
|
|
|
print(config["Account"]["email"])
|
2024-12-28 15:14:40 +08:00
|
|
|
|
return {
|
|
|
|
|
|
"account": config["Account"]["email"],
|
|
|
|
|
|
"password": config["Account"]["password"],
|
|
|
|
|
|
"first_name": config["Account"]["first_name"],
|
|
|
|
|
|
"last_name": config["Account"]["last_name"],
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-12-28 20:05:29 +08:00
|
|
|
|
raise FileNotFoundError(f"配置文件不存在: {config_path}")
|
2024-12-28 15:14:40 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def handle_turnstile(tab):
|
|
|
|
|
|
"""处理 Turnstile 验证"""
|
|
|
|
|
|
print("准备处理验证")
|
|
|
|
|
|
try:
|
|
|
|
|
|
while True:
|
|
|
|
|
|
try:
|
|
|
|
|
|
challengeCheck = (
|
|
|
|
|
|
tab.ele("@id=cf-turnstile", timeout=2)
|
|
|
|
|
|
.child()
|
|
|
|
|
|
.shadow_root.ele("tag:iframe")
|
|
|
|
|
|
.ele("tag:body")
|
|
|
|
|
|
.sr("tag:input")
|
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
if challengeCheck:
|
|
|
|
|
|
print("验证框加载完成")
|
|
|
|
|
|
time.sleep(random.uniform(1, 3))
|
|
|
|
|
|
challengeCheck.click()
|
|
|
|
|
|
print("验证按钮已点击,等待验证完成...")
|
|
|
|
|
|
time.sleep(2)
|
|
|
|
|
|
return True
|
|
|
|
|
|
except:
|
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
if tab.ele("@name=password"):
|
|
|
|
|
|
print("无需验证")
|
|
|
|
|
|
break
|
|
|
|
|
|
if tab.ele("@data-index=0"):
|
|
|
|
|
|
print("无需验证")
|
|
|
|
|
|
break
|
|
|
|
|
|
if tab.ele("Account Settings"):
|
|
|
|
|
|
print("无需验证")
|
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
|
|
time.sleep(random.uniform(1, 2))
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print(e)
|
|
|
|
|
|
print("跳过验证")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def delete_account(browser, tab):
|
|
|
|
|
|
"""删除账户流程"""
|
|
|
|
|
|
print("\n开始删除账户...")
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
if tab.ele("@name=email"):
|
|
|
|
|
|
tab.ele("@name=email").input(account)
|
|
|
|
|
|
print("输入账号")
|
|
|
|
|
|
time.sleep(random.uniform(1, 3))
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print(f"输入账号失败: {str(e)}")
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
if tab.ele("Continue"):
|
|
|
|
|
|
tab.ele("Continue").click()
|
|
|
|
|
|
print("点击Continue")
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print(f"点击Continue失败: {str(e)}")
|
|
|
|
|
|
|
|
|
|
|
|
handle_turnstile(tab)
|
|
|
|
|
|
time.sleep(5)
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
if tab.ele("@name=password"):
|
|
|
|
|
|
tab.ele("@name=password").input(password)
|
|
|
|
|
|
print("输入密码")
|
|
|
|
|
|
time.sleep(random.uniform(1, 3))
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print("输入密码失败")
|
|
|
|
|
|
|
|
|
|
|
|
sign_in_button = tab.ele(
|
|
|
|
|
|
"xpath:/html/body/div[1]/div/div/div[2]/div/form/div/button"
|
|
|
|
|
|
)
|
|
|
|
|
|
try:
|
|
|
|
|
|
if sign_in_button:
|
|
|
|
|
|
sign_in_button.click(by_js=True)
|
|
|
|
|
|
print("点击Sign in")
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print(f"点击Sign in失败: {str(e)}")
|
|
|
|
|
|
|
|
|
|
|
|
handle_turnstile(tab)
|
|
|
|
|
|
|
|
|
|
|
|
# 处理验证码
|
|
|
|
|
|
while True:
|
|
|
|
|
|
try:
|
|
|
|
|
|
if tab.ele("Account Settings"):
|
|
|
|
|
|
break
|
|
|
|
|
|
if tab.ele("@data-index=0"):
|
2024-12-28 21:56:48 +08:00
|
|
|
|
code = email_handler.get_verification_code(account)
|
2024-12-28 15:14:40 +08:00
|
|
|
|
|
|
|
|
|
|
if code:
|
|
|
|
|
|
print("获取验证码成功:", code)
|
|
|
|
|
|
else:
|
|
|
|
|
|
print("获取验证码失败,程序退出")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
i = 0
|
|
|
|
|
|
for digit in code:
|
|
|
|
|
|
tab.ele(f"@data-index={i}").input(digit)
|
|
|
|
|
|
time.sleep(random.uniform(0.1, 0.3))
|
|
|
|
|
|
i += 1
|
|
|
|
|
|
break
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print(e)
|
|
|
|
|
|
|
|
|
|
|
|
handle_turnstile(tab)
|
|
|
|
|
|
time.sleep(random.uniform(1, 3))
|
|
|
|
|
|
# tab.get_screenshot('sign-in_success.png')
|
|
|
|
|
|
# print("登录账户截图")
|
|
|
|
|
|
|
|
|
|
|
|
tab.get(settings_url)
|
|
|
|
|
|
print("进入设置页面")
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
if tab.ele("@class=mt-1"):
|
|
|
|
|
|
tab.ele("@class=mt-1").click()
|
|
|
|
|
|
print("点击Adavance")
|
|
|
|
|
|
time.sleep(random.uniform(1, 2))
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print(f"点击Adavance失败: {str(e)}")
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
if tab.ele("Delete Account"):
|
|
|
|
|
|
tab.ele("Delete Account").click()
|
|
|
|
|
|
print("点击Delete Account")
|
|
|
|
|
|
time.sleep(random.uniform(1, 2))
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print(f"点击Delete Account失败: {str(e)}")
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
if tab.ele("tag:input"):
|
|
|
|
|
|
tab.actions.click("tag:input").type("delete")
|
|
|
|
|
|
print("输入delete")
|
|
|
|
|
|
time.sleep(random.uniform(1, 2))
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print(f"输入delete失败: {str(e)}")
|
|
|
|
|
|
|
|
|
|
|
|
delete_button = tab.ele(
|
|
|
|
|
|
"xpath:/html/body/main/div/div/div/div/div/div[1]/div[2]/div[3]/div[2]/div/div/div[2]/button[2]"
|
|
|
|
|
|
)
|
|
|
|
|
|
try:
|
|
|
|
|
|
if delete_button:
|
|
|
|
|
|
print("点击Delete")
|
|
|
|
|
|
delete_button.click()
|
|
|
|
|
|
time.sleep(5)
|
|
|
|
|
|
# tab.get_screenshot('delete_account.png')
|
|
|
|
|
|
# print("删除账户截图")
|
|
|
|
|
|
return True
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print(f"点击Delete失败: {str(e)}")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def get_cursor_session_token(tab):
|
|
|
|
|
|
"""获取cursor session token"""
|
|
|
|
|
|
cookies = tab.cookies()
|
|
|
|
|
|
cursor_session_token = None
|
2024-12-28 21:56:48 +08:00
|
|
|
|
time.sleep(3)
|
|
|
|
|
|
print(cookies)
|
2024-12-28 15:14:40 +08:00
|
|
|
|
for cookie in cookies:
|
|
|
|
|
|
if cookie["name"] == "WorkosCursorSessionToken":
|
|
|
|
|
|
cursor_session_token = cookie["value"].split("%3A%3A")[1]
|
|
|
|
|
|
break
|
2024-12-28 21:56:48 +08:00
|
|
|
|
if not cursor_session_token:
|
|
|
|
|
|
print("未能获取到CursorSessionToken")
|
2024-12-28 15:14:40 +08:00
|
|
|
|
return cursor_session_token
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def update_cursor_auth(email=None, access_token=None, refresh_token=None):
|
|
|
|
|
|
"""
|
|
|
|
|
|
更新Cursor的认证信息的便捷函数
|
|
|
|
|
|
"""
|
|
|
|
|
|
auth_manager = CursorAuthManager()
|
|
|
|
|
|
return auth_manager.update_auth(email, access_token, refresh_token)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def sign_up_account(browser, tab):
|
|
|
|
|
|
"""注册账户流程"""
|
|
|
|
|
|
print("\n开始注册新账户...")
|
|
|
|
|
|
tab.get(sign_up_url)
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
if tab.ele("@name=first_name"):
|
|
|
|
|
|
print("已打开注册页面")
|
|
|
|
|
|
tab.actions.click("@name=first_name").input(first_name)
|
|
|
|
|
|
time.sleep(random.uniform(1, 3))
|
|
|
|
|
|
|
|
|
|
|
|
tab.actions.click("@name=last_name").input(last_name)
|
|
|
|
|
|
time.sleep(random.uniform(1, 3))
|
|
|
|
|
|
|
|
|
|
|
|
tab.actions.click("@name=email").input(account)
|
|
|
|
|
|
print("输入邮箱")
|
|
|
|
|
|
time.sleep(random.uniform(1, 3))
|
|
|
|
|
|
|
|
|
|
|
|
tab.actions.click("@type=submit")
|
|
|
|
|
|
print("点击注册按钮")
|
|
|
|
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print("打开注册页面失败")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
handle_turnstile(tab)
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
|
if tab.ele("@name=password"):
|
|
|
|
|
|
tab.ele("@name=password").input(password)
|
|
|
|
|
|
print("输入密码")
|
|
|
|
|
|
time.sleep(random.uniform(1, 3))
|
|
|
|
|
|
|
|
|
|
|
|
tab.ele("@type=submit").click()
|
|
|
|
|
|
print("点击Continue按钮")
|
|
|
|
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print("输入密码失败")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
time.sleep(random.uniform(1, 3))
|
|
|
|
|
|
if tab.ele("This email is not available."):
|
|
|
|
|
|
print("This email is not available.")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
handle_turnstile(tab)
|
|
|
|
|
|
|
|
|
|
|
|
while True:
|
|
|
|
|
|
try:
|
|
|
|
|
|
if tab.ele("Account Settings"):
|
|
|
|
|
|
break
|
|
|
|
|
|
if tab.ele("@data-index=0"):
|
2024-12-28 21:56:48 +08:00
|
|
|
|
code = email_handler.get_verification_code(account)
|
2024-12-28 15:14:40 +08:00
|
|
|
|
|
|
|
|
|
|
if code:
|
|
|
|
|
|
print("获取验证码成功:", code)
|
|
|
|
|
|
else:
|
|
|
|
|
|
print("获取验证码失败,程序退出")
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
i = 0
|
|
|
|
|
|
for digit in code:
|
|
|
|
|
|
tab.ele(f"@data-index={i}").input(digit)
|
|
|
|
|
|
time.sleep(random.uniform(0.1, 0.3))
|
|
|
|
|
|
i += 1
|
|
|
|
|
|
break
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print(e)
|
|
|
|
|
|
|
|
|
|
|
|
handle_turnstile(tab)
|
|
|
|
|
|
|
|
|
|
|
|
time.sleep(random.uniform(1, 3))
|
|
|
|
|
|
print("进入设置页面")
|
|
|
|
|
|
tab.get(settings_url)
|
|
|
|
|
|
try:
|
|
|
|
|
|
usage_ele = tab.ele(
|
|
|
|
|
|
"xpath:/html/body/main/div/div/div/div/div/div[2]/div/div/div/div[1]/div[1]/span[2]"
|
|
|
|
|
|
)
|
|
|
|
|
|
if usage_ele:
|
|
|
|
|
|
usage_info = usage_ele.text
|
|
|
|
|
|
total_usage = usage_info.split("/")[-1].strip()
|
|
|
|
|
|
print("可用上限: " + total_usage)
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
print("获取可用上限失败")
|
|
|
|
|
|
# tab.get_screenshot("sign_up_success.png")
|
|
|
|
|
|
# print("注册账户截图")
|
|
|
|
|
|
print("注册完成")
|
|
|
|
|
|
print("Cursor 账号: " + account)
|
|
|
|
|
|
print(" 密码: " + password)
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-12-28 20:05:29 +08:00
|
|
|
|
def cleanup_temp_files():
|
|
|
|
|
|
"""清理临时文件和缓存"""
|
|
|
|
|
|
try:
|
|
|
|
|
|
temp_dirs = [
|
2024-12-28 20:34:34 +08:00
|
|
|
|
os.path.join(os.getcwd(), "__pycache__"),
|
|
|
|
|
|
os.path.join(os.getcwd(), "build"),
|
2024-12-28 20:05:29 +08:00
|
|
|
|
]
|
2024-12-28 20:34:34 +08:00
|
|
|
|
|
2024-12-28 20:05:29 +08:00
|
|
|
|
for dir_path in temp_dirs:
|
|
|
|
|
|
if os.path.exists(dir_path):
|
|
|
|
|
|
import shutil
|
2024-12-28 20:34:34 +08:00
|
|
|
|
|
2024-12-28 20:05:29 +08:00
|
|
|
|
shutil.rmtree(dir_path)
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
|
logging.warning(f"清理临时文件失败: {str(e)}")
|
|
|
|
|
|
|
|
|
|
|
|
|
2024-12-28 15:14:40 +08:00
|
|
|
|
if __name__ == "__main__":
|
2024-12-28 21:56:48 +08:00
|
|
|
|
browser_manager = None
|
2024-12-28 15:14:40 +08:00
|
|
|
|
try:
|
|
|
|
|
|
# 加载配置
|
|
|
|
|
|
config = load_config()
|
|
|
|
|
|
|
2024-12-28 21:56:48 +08:00
|
|
|
|
# 初始化浏览器
|
|
|
|
|
|
browser_manager = BrowserManager()
|
|
|
|
|
|
browser = browser_manager.init_browser()
|
|
|
|
|
|
|
|
|
|
|
|
# 初始化<E5A78B><E58C96>箱验证处理器
|
|
|
|
|
|
email_handler = EmailVerificationHandler(browser)
|
|
|
|
|
|
|
2024-12-28 15:14:40 +08:00
|
|
|
|
# 固定的 URL 配置
|
|
|
|
|
|
login_url = "https://authenticator.cursor.sh"
|
|
|
|
|
|
sign_up_url = "https://authenticator.cursor.sh/sign-up"
|
|
|
|
|
|
settings_url = "https://www.cursor.com/settings"
|
|
|
|
|
|
mail_url = "https://tempmail.plus"
|
|
|
|
|
|
|
|
|
|
|
|
# 账号信息
|
|
|
|
|
|
account = config["account"]
|
|
|
|
|
|
password = config["password"]
|
|
|
|
|
|
first_name = config["first_name"]
|
|
|
|
|
|
last_name = config["last_name"]
|
|
|
|
|
|
|
|
|
|
|
|
auto_update_cursor_auth = True
|
|
|
|
|
|
|
|
|
|
|
|
tab = browser.latest_tab
|
|
|
|
|
|
tab.run_js("try { turnstile.reset() } catch(e) { }")
|
|
|
|
|
|
|
|
|
|
|
|
print("开始执行删除和注册流程")
|
|
|
|
|
|
print("***请确认已经用https://tempmail.plus/zh邮箱成功申请过cursor账号!***")
|
|
|
|
|
|
tab.get(login_url)
|
|
|
|
|
|
|
|
|
|
|
|
# 执行删除和注册流程
|
|
|
|
|
|
if delete_account(browser, tab):
|
|
|
|
|
|
print("账户删除成功")
|
|
|
|
|
|
time.sleep(3)
|
|
|
|
|
|
if sign_up_account(browser, tab):
|
|
|
|
|
|
token = get_cursor_session_token(tab)
|
2024-12-28 21:56:48 +08:00
|
|
|
|
if token:
|
|
|
|
|
|
print(f"CursorSessionToken: {token}")
|
|
|
|
|
|
print("账户注册成功")
|
|
|
|
|
|
if auto_update_cursor_auth:
|
|
|
|
|
|
update_cursor_auth(
|
|
|
|
|
|
email=account, access_token=token, refresh_token=token
|
|
|
|
|
|
)
|
|
|
|
|
|
else:
|
|
|
|
|
|
print("未能获取到CursorSessionToken")
|
2024-12-28 15:14:40 +08:00
|
|
|
|
else:
|
|
|
|
|
|
print("账户注册失败")
|
|
|
|
|
|
else:
|
|
|
|
|
|
print("账户删除失败")
|
2024-12-28 21:56:48 +08:00
|
|
|
|
# if sign_up_account(browser, tab):
|
|
|
|
|
|
# token = get_cursor_session_token(tab)
|
|
|
|
|
|
# print(f"CursorSessionToken: {token}")
|
|
|
|
|
|
# print("账户注册成功")
|
|
|
|
|
|
# if auto_update_cursor_auth:
|
|
|
|
|
|
# update_cursor_auth(
|
|
|
|
|
|
# email=account, access_token=token, refresh_token=token
|
|
|
|
|
|
# )
|
|
|
|
|
|
# else:
|
|
|
|
|
|
# print("账户注册失败")
|
2024-12-28 15:14:40 +08:00
|
|
|
|
|
|
|
|
|
|
print("脚本执行完毕")
|
|
|
|
|
|
|
|
|
|
|
|
except Exception as e:
|
2024-12-28 21:56:48 +08:00
|
|
|
|
logging.error(f"程序执行出错: {str(e)}")
|
2024-12-28 20:05:29 +08:00
|
|
|
|
import traceback
|
2024-12-28 20:34:34 +08:00
|
|
|
|
|
2024-12-28 21:56:48 +08:00
|
|
|
|
logging.error(traceback.format_exc())
|
2024-12-28 20:05:29 +08:00
|
|
|
|
finally:
|
2024-12-28 21:56:48 +08:00
|
|
|
|
if browser_manager:
|
|
|
|
|
|
browser_manager.quit()
|
2024-12-28 15:14:40 +08:00
|
|
|
|
input("\n按回车键退出...")
|