random_proxy.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  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. ["52001", "53001"],
  15. ["52002", "53002"],
  16. ["52003", "53003"],
  17. ["52004", "53004"],
  18. ["52005", "53005"],
  19. ["52006", "53006"],
  20. ["52007", "53007"],
  21. ["52008", "53008"],
  22. ["52009", "53009"],
  23. ["52011", "53012"],
  24. ["52012", "53012"]
  25. ]
  26. class ClashProxyManager:
  27. def __init__(self, base_url, base_port):
  28. self.key_group = 0
  29. self.base_url = base_url
  30. self.base_port = base_port
  31. self.all_proxies = []
  32. def get_all_proxies(self, clash_tool_url: str) -> List[str]:
  33. url = f"{clash_tool_url}/api/proxies"
  34. try:
  35. response = httpx.get(url)
  36. response.raise_for_status()
  37. proxies = response.json()
  38. logging.info("Available proxies:")
  39. # 输出读取的所有代理信息
  40. # for proxy_name, proxy_info in proxies['proxies'].items():
  41. # logging.info(f"Name: {proxy_name}, Type: {proxy_info.get('type', 'Unknown')}")
  42. proxy_list = list(proxies['proxies'].keys())
  43. filtered_list = [item for item in proxy_list if item not in {'REJECT', 'GLOBAL', 'DIRECT'}]
  44. return filtered_list
  45. except Exception as e:
  46. logging.error(f"Failed to get proxies: {e}")
  47. return []
  48. def switch_proxy(self, proxy_name: str, url_and_port: str) -> None:
  49. logging.info("switch proxy")
  50. url = f"{url_and_port}/api/proxies/GLOBAL"
  51. data = {"name": proxy_name}
  52. try:
  53. response = httpx.put(url, json=data)
  54. if response.status_code == 204:
  55. logging.info(f"Switched to proxy: {proxy_name}")
  56. else:
  57. logging.error(f"Failed to switch proxy: {response.status_code} - {proxy_name}")
  58. except Exception as e:
  59. logging.error(f"Failed to switch proxy: {e}")
  60. def update_configs(self):
  61. for base_port in self.base_port:
  62. url_and_port = self.base_url + ":" + base_port[0]
  63. key = "/api/configs"
  64. url = url_and_port + key
  65. headers = {
  66. "accept": "application/json, text/plain, */*",
  67. "accept-encoding": "gzip, deflate, br, zstd",
  68. "accept-language": "zh-CN,zh",
  69. "connection": "keep-alive",
  70. "content-type": "application/json",
  71. "origin": url_and_port,
  72. "referer": url_and_port,
  73. "sec-ch-ua": '"Not A(Brand";v="8", "Chromium";v="132", "Brave";v="132"',
  74. "sec-ch-ua-mobile": "?0",
  75. "sec-ch-ua-platform": '"macOS"',
  76. "sec-fetch-dest": "empty",
  77. "sec-fetch-mode": "cors",
  78. "sec-fetch-site": "same-origin",
  79. "sec-gpc": "1",
  80. "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",
  81. }
  82. # 请求体数据
  83. data = {"mode": "Global"} # 替换为实际的请求数据
  84. # 使用 httpx 发送 PATCH 请求
  85. try:
  86. with httpx.Client() as client:
  87. response = client.patch(url, headers=headers, json=data)
  88. if response.status_code == 204:
  89. print(f"{url} OK")
  90. else:
  91. print("响应内容:", response.text)
  92. except httpx.RequestError as exc:
  93. print(f"请求失败: {exc}")
  94. def check_proxy(self, proxy_url):
  95. # proxy_url: 代理地址, 没有密码
  96. # 测试目标地址:
  97. command = ["curl", "-x", proxy_url, "ip.sb"]
  98. # 执行命令并获取输出
  99. try:
  100. result = subprocess.run(command, capture_output=True, text=True, check=True)
  101. # 打印命令的标准输出
  102. print("Output:", result.stdout)
  103. return True
  104. except subprocess.CalledProcessError as e:
  105. # 如果命令执行失败,打印错误信息
  106. print("Error:", e.stderr)
  107. return False
  108. def main(self) -> None:
  109. # 设置全局代理
  110. self.update_configs()
  111. # 读取所有代理
  112. if not self.all_proxies:
  113. for port_list in self.base_port:
  114. base_url = self.base_url + ":" + port_list[0]
  115. clash_tool_url = f"{base_url}"
  116. proxies = self.get_all_proxies(clash_tool_url)
  117. if proxies:
  118. self.all_proxies = proxies
  119. break
  120. if not self.all_proxies:
  121. logging.error("Failed to get all proxies")
  122. return
  123. else:
  124. logging.info(f"Found {self.all_proxies} proxies.")
  125. logging.info(f"Found {len(self.all_proxies)} proxies.")
  126. # 遍历所有的线路api, 切换不重复代理
  127. # 切换后, 检测代理, 如果检测返回失败, 再次切换
  128. used_proxy = []
  129. for base_port in self.base_port:
  130. url_and_port = self.base_url + ":" + base_port[0]
  131. proxy_url = self.base_url + ":" + base_port[1]
  132. while True:
  133. # 选一个代理
  134. choose_proxy = random.choice(self.all_proxies)
  135. # 如果已经选过就继续下一个
  136. if choose_proxy in used_proxy:
  137. continue
  138. # 切换代理
  139. self.switch_proxy(choose_proxy, url_and_port)
  140. # 切换后检测代理
  141. if self.check_proxy(proxy_url):
  142. print(f"代理 {choose_proxy} 切换成功,检测通过!")
  143. used_proxy.append(choose_proxy) # 标记为已使用
  144. break # 成功后退出当前代理的重试循环
  145. else:
  146. print(f"{url_and_port} 切换 {choose_proxy} 检测失败")
  147. time.sleep(1)
  148. if __name__ == "__main__":
  149. manager = ClashProxyManager(BASE_URL, PORT_LIST)
  150. manager.main()