switch_random_proxy_web3.py 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. # -*- coding: utf-8 -*-
  2. '''
  3. 切换到随机代理
  4. '''
  5. import random
  6. from urllib.parse import quote
  7. import httpx
  8. import time
  9. import subprocess
  10. from typing import List
  11. BASE_URL = "http://192.168.31.194"
  12. PORT_LIST = ['58001', '58002', '58003', '58004', '58005', '58006', '58007', '58008', '58009', '58010']
  13. GROUP = 1
  14. class ClashProxyManager:
  15. def __init__(self, base_url, base_port):
  16. self.key_group = 0
  17. self.base_url = base_url
  18. self.base_port = base_port
  19. self.all_proxies = []
  20. self.used_proxy = []
  21. def get_all_proxies(self, clash_tool_url: str) -> List[str]:
  22. # 连接其中一个代理服务器, 获取所有代理节点, 因为每个代理服务器的所有节点都是一样的, 所以获取一个就行了
  23. url = f"{clash_tool_url}/api/proxies"
  24. try:
  25. response = httpx.get(url)
  26. response.raise_for_status()
  27. proxies = response.json()
  28. # 输出读取的所有代理信息
  29. # for proxy_name, proxy_info in proxies['proxies'].items():
  30. # logging.info(f"Name: {proxy_name}, Type: {proxy_info.get('type', 'Unknown')}")
  31. proxy_list = list(proxies['proxies'].keys())
  32. result_data = []
  33. if GROUP == 1:
  34. for filtered in proxy_list:
  35. for group in PROXY_GROUP:
  36. if group.lower() in filtered.lower() and filtered not in {'REJECT', 'GLOBAL', 'DIRECT'}:
  37. result_data.append(filtered)
  38. elif GROUP == 2:
  39. for filtered in proxy_list:
  40. if all(group.lower() not in filtered.lower() for group in PROXY_GROUP) and filtered not in {
  41. 'REJECT', 'GLOBAL', 'DIRECT'}:
  42. result_data.append(filtered)
  43. else:
  44. print(f'选择代理组为 {GROUP}')
  45. exit(1)
  46. print(f'读取所有代理节点, 一共 {len(result_data)} 个节点')
  47. return result_data
  48. except Exception as e:
  49. print(f"Failed to get proxies: {e}")
  50. return []
  51. def switch_proxy(self, proxy_name: str, url_and_port: str) -> None:
  52. # 根据节点名称和代理服务器url, 切换到指定的代理节点
  53. url = f"{url_and_port}/api/proxies/GLOBAL"
  54. data = {"name": proxy_name}
  55. try:
  56. response = httpx.put(url, json=data)
  57. if response.status_code == 204:
  58. print(f"Switched to proxy: {proxy_name}")
  59. else:
  60. print(f"Failed to switch proxy: {response.status_code} - {proxy_name}")
  61. except Exception as e:
  62. print(f"Failed to switch proxy: {e}")
  63. def update_configs(self):
  64. for base_port in self.base_port:
  65. url_and_port = self.base_url + ":" + base_port
  66. key = "/api/configs"
  67. url = url_and_port + key
  68. # 构造 curl 命令
  69. curl_cmd = [
  70. "curl",
  71. "-X", "PATCH",
  72. url,
  73. "-H", "Content-Type: application/json",
  74. "-d", '{"mode": "Global"}'
  75. ]
  76. try:
  77. # 使用 subprocess 运行 curl 命令
  78. result = subprocess.run(curl_cmd, capture_output=True, text=True, check=True)
  79. # 检查 curl 命令的返回状态
  80. if result.returncode != 0:
  81. print(f"请求失败,状态码: {result.returncode}")
  82. print("错误信息:", result.stderr.strip())
  83. except subprocess.CalledProcessError as exc:
  84. print(f"请求失败: {exc}")
  85. print("错误信息:", exc.stderr.strip())
  86. def check_proxy(self, proxy_url, choose_proxy):
  87. encode_proxy_name = quote(choose_proxy, safe="")
  88. command = [
  89. "curl",
  90. "-X", "GET",
  91. f"{proxy_url}/api/proxies/{encode_proxy_name}/delay?timeout=5000&url=http:%2F%2Fwww.gstatic.com%2Fgenerate_204"
  92. ]
  93. try:
  94. result = subprocess.run(command, capture_output=True, text=True, check=True)
  95. print("Output:", result.stdout)
  96. if 'Timeout' in result.stdout:
  97. return False
  98. return True
  99. except subprocess.CalledProcessError as e:
  100. print("Error:", e.stderr)
  101. return False
  102. def main(self) -> None:
  103. self.update_configs()
  104. if not self.all_proxies:
  105. for port_list in self.base_port:
  106. base_url = self.base_url + ":" + port_list
  107. clash_tool_url = f"{base_url}"
  108. self.all_proxies = self.get_all_proxies(clash_tool_url)
  109. if self.all_proxies:
  110. break
  111. if not self.all_proxies:
  112. return
  113. else:
  114. print(f'待切换节点一共 {len(self.all_proxies)} 个')
  115. for base_port in self.base_port:
  116. url_and_port = self.base_url + ":" + base_port
  117. while True:
  118. choose_proxy = random.choice(self.all_proxies)
  119. if choose_proxy in self.used_proxy:
  120. if len(self.used_proxy) >= len(self.all_proxies):
  121. print('=' * 88)
  122. print(f'所有节点已尝试过, 退出')
  123. print('=' * 88)
  124. return
  125. continue
  126. self.switch_proxy(choose_proxy, url_and_port)
  127. self.used_proxy.append(choose_proxy)
  128. if self.check_proxy(url_and_port, choose_proxy):
  129. print(f"代理 {choose_proxy} 切换成功,检测通过!")
  130. print('=' * 88)
  131. break
  132. else:
  133. print(f"{url_and_port} 切换 xxxxxxxx {choose_proxy} xxxxxxxx 检测失败")
  134. time.sleep(1)
  135. if __name__ == "__main__":
  136. manager = ClashProxyManager(BASE_URL, PORT_LIST)
  137. manager.main()