set_global_switch_proxy_random.py 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169
  1. # -*- coding: utf-8 -*-
  2. '''
  3. 切换到随机代理
  4. '''
  5. import random
  6. import httpx
  7. import time
  8. import logging
  9. import subprocess
  10. from typing import Optional, List
  11. logging.basicConfig(level=logging.INFO)
  12. BASE_URL = "http://192.168.31.201"
  13. PORT_LIST = [
  14. ["58001", "59001"],
  15. ["58002", "59002"],
  16. ["58003", "59003"],
  17. ["58004", "59004"],
  18. ["58005", "59005"],
  19. ["58006", "59006"],
  20. ["58007", "59007"],
  21. ["58008", "59008"],
  22. ["58009", "59009"],
  23. ["58010", "59010"],
  24. ]
  25. TEST_URL = "https://httpbin.org/ip"
  26. class ClashProxyManager:
  27. def __init__(self, base_url, base_port, test_url):
  28. self.key_group = 0
  29. self.base_url = base_url
  30. self.base_port = base_port
  31. self.test_url = test_url
  32. self.all_proxies = []
  33. def get_all_proxies(self, clash_tool_url: str) -> List[str]:
  34. url = f"{clash_tool_url}/api/proxies"
  35. try:
  36. response = httpx.get(url)
  37. response.raise_for_status()
  38. proxies = response.json()
  39. logging.info("Available proxies:")
  40. # 输出读取的所有代理信息
  41. # for proxy_name, proxy_info in proxies['proxies'].items():
  42. # logging.info(f"Name: {proxy_name}, Type: {proxy_info.get('type', 'Unknown')}")
  43. return list(proxies['proxies'].keys())
  44. except Exception as e:
  45. logging.error(f"Failed to get proxies: {e}")
  46. return []
  47. def switch_proxy(self, proxy_name: str, url_and_port: str) -> None:
  48. logging.info("switch proxy")
  49. url = f"{url_and_port}/api/proxies/GLOBAL"
  50. data = {"name": proxy_name}
  51. try:
  52. response = httpx.put(url, json=data)
  53. if response.status_code == 204:
  54. logging.info(f"Switched to proxy: {proxy_name}")
  55. else:
  56. logging.error(f"Failed to switch proxy: {response.status_code} - {proxy_name}")
  57. except Exception as e:
  58. logging.error(f"Failed to switch proxy: {e}")
  59. def update_configs(self):
  60. for base_port in self.base_port:
  61. url_and_port = self.base_url + ":" + base_port[0]
  62. key = "/api/configs"
  63. url = url_and_port + key
  64. headers = {
  65. "accept": "application/json, text/plain, */*",
  66. "accept-encoding": "gzip, deflate, br, zstd",
  67. "accept-language": "zh-CN,zh",
  68. "connection": "keep-alive",
  69. "content-type": "application/json",
  70. "origin": url_and_port,
  71. "referer": url_and_port,
  72. "sec-ch-ua": '"Not A(Brand";v="8", "Chromium";v="132", "Brave";v="132"',
  73. "sec-ch-ua-mobile": "?0",
  74. "sec-ch-ua-platform": '"macOS"',
  75. "sec-fetch-dest": "empty",
  76. "sec-fetch-mode": "cors",
  77. "sec-fetch-site": "same-origin",
  78. "sec-gpc": "1",
  79. "user-agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/132.0.0.0 Safari/537.36",
  80. }
  81. # 请求体数据
  82. data = {"mode": "Global"} # 替换为实际的请求数据
  83. # 使用 httpx 发送 PATCH 请求
  84. try:
  85. with httpx.Client() as client:
  86. response = client.patch(url, headers=headers, json=data)
  87. if response.status_code == 204:
  88. print(f"{url} OK")
  89. else:
  90. print("响应内容:", response.text)
  91. except httpx.RequestError as exc:
  92. print(f"请求失败: {exc}")
  93. def check_proxy(self, proxy_url):
  94. # proxy_url: 代理地址, 没有密码
  95. # 测试目标地址:
  96. command = ["curl", "-x", proxy_url, "ip.sb"]
  97. # 执行命令并获取输出
  98. try:
  99. result = subprocess.run(command, capture_output=True, text=True, check=True)
  100. # 打印命令的标准输出
  101. print("Output:", result.stdout)
  102. return True
  103. except subprocess.CalledProcessError as e:
  104. # 如果命令执行失败,打印错误信息
  105. print("Error:", e.stderr)
  106. return False
  107. def main(self) -> None:
  108. # 设置全局代理
  109. self.update_configs()
  110. # 读取所有代理
  111. if not self.all_proxies:
  112. for port_list in self.base_port:
  113. base_url = self.base_url + ":" + port_list[0]
  114. clash_tool_url = f"{base_url}"
  115. proxies = self.get_all_proxies(clash_tool_url)
  116. if proxies:
  117. self.all_proxies = proxies
  118. break
  119. if not self.all_proxies:
  120. logging.error("Failed to get all proxies")
  121. return
  122. # 遍历所有的线路api, 切换不重复代理
  123. # 切换后, 检测代理, 如果检测返回失败, 再次切换
  124. used_proxy = []
  125. for base_port in self.base_port:
  126. url_and_port = self.base_url + ":" + base_port[0]
  127. proxy_url = self.base_url + ":" + base_port[1]
  128. while True:
  129. # 选一个代理
  130. choose_proxy = random.choice(self.all_proxies)
  131. # 如果已经选过就继续下一个
  132. if choose_proxy in used_proxy:
  133. continue
  134. # 切换代理
  135. self.switch_proxy(choose_proxy, url_and_port)
  136. # 切换后检测代理
  137. if self.check_proxy(proxy_url):
  138. print(f"代理 {choose_proxy} 切换成功,检测通过!")
  139. used_proxy.append(choose_proxy) # 标记为已使用
  140. break # 成功后退出当前代理的重试循环
  141. else:
  142. print(f"{url_and_port} 切换 {choose_proxy} 检测失败")
  143. time.sleep(1)
  144. if __name__ == "__main__":
  145. manager = ClashProxyManager(BASE_URL, PORT_LIST, TEST_URL)
  146. manager.main()