gui.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. """A simple tkinter GUI to edit config.json and manage protector list.
  2. Features:
  3. - Edit base_url, conn_threshold, scan_interval (saved to config.json)
  4. - Manage protector_list entries (add/edit/remove) stored in protector_list.json
  5. - On saving base_url, attempt login via auth_session.login() and store sess_key
  6. """
  7. from __future__ import annotations
  8. import json
  9. import threading
  10. import tkinter as tk
  11. from tkinter import messagebox
  12. from pathlib import Path
  13. from auth_session import load_config, get_base_url, login, set_sess_key
  14. import protector_list
  15. ROOT = Path(__file__).resolve().parent
  16. CONFIG_PATH = ROOT / "config.json"
  17. def load_cfg():
  18. return load_config()
  19. def save_cfg(cfg: dict):
  20. with open(CONFIG_PATH, "w", encoding="utf-8") as f:
  21. json.dump(cfg, f, ensure_ascii=False, indent=2)
  22. class ProtectorGUI(tk.Tk):
  23. def __init__(self, protector_runner=None):
  24. super().__init__()
  25. self.title("Protector GUI")
  26. self.geometry("700x500")
  27. self.cfg = load_cfg()
  28. # Config frame
  29. cf = tk.LabelFrame(self, text="Config")
  30. cf.pack(fill="x", padx=8, pady=6)
  31. tk.Label(cf, text="base_url:").grid(row=0, column=0, sticky="w")
  32. self.base_entry = tk.Entry(cf, width=60)
  33. self.base_entry.grid(row=0, column=1, padx=4, pady=2)
  34. tk.Label(cf, text="conn_threshold:").grid(row=1, column=0, sticky="w")
  35. self.th_entry = tk.Entry(cf, width=10)
  36. self.th_entry.grid(row=1, column=1, sticky="w", padx=4, pady=2)
  37. tk.Label(cf, text="test_prefix:").grid(row=1, column=2, sticky="w", padx=(12,0))
  38. self.test_prefix_entry = tk.Entry(cf, width=15)
  39. self.test_prefix_entry.grid(row=1, column=3, sticky="w", padx=4, pady=2)
  40. tk.Label(cf, text="rule_ip_limit:").grid(row=2, column=2, sticky="w", padx=(12,0))
  41. self.rule_limit_entry = tk.Entry(cf, width=15)
  42. self.rule_limit_entry.grid(row=2, column=3, sticky="w", padx=4, pady=2)
  43. tk.Label(cf, text="scan_interval (s):").grid(row=2, column=0, sticky="w")
  44. self.interval_entry = tk.Entry(cf, width=10)
  45. self.interval_entry.grid(row=2, column=1, sticky="w", padx=4, pady=2)
  46. tk.Button(cf, text="Save Config", command=self.save_config).grid(row=3, column=1, sticky="w", pady=6)
  47. # Protector list frame
  48. lf = tk.LabelFrame(self, text="Protector List")
  49. lf.pack(fill="both", expand=True, padx=8, pady=6)
  50. self.listbox = tk.Listbox(lf)
  51. self.listbox.pack(side="left", fill="both", expand=True, padx=4, pady=4)
  52. self.listbox.bind("<<ListboxSelect>>", self.on_select)
  53. right = tk.Frame(lf)
  54. right.pack(side="right", fill="y", padx=4)
  55. tk.Label(right, text="target_ip:").pack(anchor="w")
  56. self.ip_entry = tk.Entry(right)
  57. self.ip_entry.pack(fill="x")
  58. tk.Label(right, text="src_port:").pack(anchor="w")
  59. self.port_entry = tk.Entry(right)
  60. self.port_entry.pack(fill="x")
  61. tk.Label(right, text="threshold (optional):").pack(anchor="w")
  62. self.entry_threshold = tk.Entry(right)
  63. self.entry_threshold.pack(fill="x")
  64. tk.Button(right, text="Add", command=self.add_entry).pack(fill="x", pady=4)
  65. tk.Button(right, text="Update", command=self.update_entry).pack(fill="x", pady=4)
  66. tk.Button(right, text="Remove", command=self.remove_entry).pack(fill="x", pady=4)
  67. self.load_values()
  68. # 自动在后台尝试登录(启动后立即)
  69. def _auto_login():
  70. try:
  71. resp, sess_cookie = login()
  72. if sess_cookie:
  73. set_sess_key(sess_cookie)
  74. # 不弹出大量通知,只有在需要时可打开下面这一行
  75. # messagebox.showinfo("登录", f"自动登录完成, 状态: {resp.status_code}")
  76. except Exception as e:
  77. # 登录失败不阻塞 GUI,记录到控制台/弹出一条错误提示
  78. try:
  79. messagebox.showerror("自动登录失败", str(e))
  80. except Exception:
  81. print("自动登录失败:", e)
  82. threading.Thread(target=_auto_login, daemon=True).start()
  83. # Protector runner instance (optional) for countdown and triggering runs
  84. self.protector_runner = protector_runner
  85. # countdown label
  86. self.countdown_var = tk.StringVar(value="Protector: idle")
  87. self.countdown_label = tk.Label(self, textvariable=self.countdown_var)
  88. self.countdown_label.pack(anchor="ne", padx=8, pady=4)
  89. # start initial delayed run countdown (10 seconds) if runner provided
  90. if self.protector_runner is not None:
  91. self._initial_delay = 10
  92. self._countdown_seconds = self._initial_delay
  93. # schedule countdown update every 1 second via tkinter's after
  94. self.after(1000, self._update_countdown)
  95. def load_values(self):
  96. self.cfg = load_cfg()
  97. self.base_entry.delete(0, tk.END)
  98. self.base_entry.insert(0, self.cfg.get("base_url", ""))
  99. self.th_entry.delete(0, tk.END)
  100. self.th_entry.insert(0, str(self.cfg.get("conn_threshold", "")))
  101. # new fields
  102. self.test_prefix_entry.delete(0, tk.END)
  103. self.test_prefix_entry.insert(0, str(self.cfg.get("test_prefix", "Test_")))
  104. self.rule_limit_entry.delete(0, tk.END)
  105. self.rule_limit_entry.insert(0, str(self.cfg.get("rule_ip_limit", 1000)))
  106. self.interval_entry.delete(0, tk.END)
  107. self.interval_entry.insert(0, str(self.cfg.get("scan_interval", 60)))
  108. self.reload_listbox()
  109. def reload_listbox(self):
  110. self.listbox.delete(0, tk.END)
  111. items = protector_list.load_list()
  112. for i in items:
  113. self.listbox.insert(tk.END, f"{i['id']}: {i['target_ip']}:{i['src_port']} (th={i.get('threshold')})")
  114. def save_config(self):
  115. try:
  116. self.cfg["base_url"] = self.base_entry.get().strip()
  117. self.cfg["conn_threshold"] = int(self.th_entry.get().strip())
  118. # optional new fields
  119. self.cfg["test_prefix"] = str(self.test_prefix_entry.get().strip())
  120. self.cfg["rule_ip_limit"] = int(self.rule_limit_entry.get().strip())
  121. self.cfg["scan_interval"] = int(self.interval_entry.get().strip())
  122. except Exception as e:
  123. messagebox.showerror("错误", f"配置输入无效: {e}")
  124. return
  125. save_cfg(self.cfg)
  126. # try login on a background thread
  127. def _login():
  128. try:
  129. resp, sess_cookie = login()
  130. if sess_cookie:
  131. set_sess_key(sess_cookie)
  132. messagebox.showinfo("登录", f"登录完成, 状态: {resp.status_code}")
  133. except Exception as e:
  134. messagebox.showerror("登录失败", str(e))
  135. threading.Thread(target=_login, daemon=True).start()
  136. def on_select(self, evt):
  137. sel = self.listbox.curselection()
  138. if not sel:
  139. return
  140. idx = sel[0]
  141. items = protector_list.load_list()
  142. if idx >= len(items):
  143. return
  144. item = items[idx]
  145. self.ip_entry.delete(0, tk.END)
  146. self.ip_entry.insert(0, item.get("target_ip", ""))
  147. self.port_entry.delete(0, tk.END)
  148. self.port_entry.insert(0, str(item.get("src_port", "")))
  149. self.entry_threshold.delete(0, tk.END)
  150. self.entry_threshold.insert(0, str(item.get("threshold", "")))
  151. def add_entry(self):
  152. ip = self.ip_entry.get().strip()
  153. port = self.port_entry.get().strip()
  154. th = self.entry_threshold.get().strip() or None
  155. if not ip or not port:
  156. messagebox.showerror("错误", "ip 与 port 为必填")
  157. return
  158. try:
  159. entry = protector_list.add_entry(ip, int(port), int(th) if th else None)
  160. self.reload_listbox()
  161. messagebox.showinfo("添加", f"已添加: {entry}")
  162. except Exception as e:
  163. messagebox.showerror("错误", str(e))
  164. def update_entry(self):
  165. sel = self.listbox.curselection()
  166. if not sel:
  167. messagebox.showerror("错误", "先选择一条")
  168. return
  169. idx = sel[0]
  170. items = protector_list.load_list()
  171. if idx >= len(items):
  172. return
  173. eid = items[idx]["id"]
  174. ip = self.ip_entry.get().strip()
  175. port = self.port_entry.get().strip()
  176. th = self.entry_threshold.get().strip() or None
  177. try:
  178. updated = protector_list.update_entry(eid, target_ip=ip, src_port=int(port), threshold=(int(th) if th else None))
  179. if updated is None:
  180. messagebox.showerror("错误", "更新失败")
  181. else:
  182. self.reload_listbox()
  183. messagebox.showinfo("更新", f"已更新: {updated}")
  184. except Exception as e:
  185. messagebox.showerror("错误", str(e))
  186. def remove_entry(self):
  187. sel = self.listbox.curselection()
  188. if not sel:
  189. messagebox.showerror("错误", "先选择一条")
  190. return
  191. idx = sel[0]
  192. items = protector_list.load_list()
  193. if idx >= len(items):
  194. return
  195. eid = items[idx]["id"]
  196. ok = protector_list.remove_entry(eid)
  197. if ok:
  198. self.reload_listbox()
  199. messagebox.showinfo("删除", "已删除")
  200. else:
  201. messagebox.showerror("错误", "删除失败")
  202. def _update_countdown(self):
  203. """Update countdown label every second and trigger protector run when it reaches zero."""
  204. try:
  205. if not self.protector_runner:
  206. return
  207. # Update label
  208. try:
  209. self.countdown_var.set(f"下一次检测:{self._countdown_seconds}s")
  210. except Exception:
  211. self.countdown_var.set("Protector: running")
  212. if self._countdown_seconds <= 0:
  213. # trigger a run in background
  214. try:
  215. threading.Thread(target=self.protector_runner.run_once, daemon=True).start()
  216. except Exception as e:
  217. print("触发 protector 运行失败:", e)
  218. # reset countdown to configured interval
  219. try:
  220. self._countdown_seconds = int(self.protector_runner.get_interval())
  221. except Exception:
  222. self._countdown_seconds = self._initial_delay
  223. else:
  224. self._countdown_seconds -= 1
  225. except Exception as e:
  226. # keep ticking even on error
  227. print("倒计时更新出错:", e)
  228. # schedule next tick
  229. try:
  230. self.after(1000, self._update_countdown)
  231. except Exception:
  232. pass
  233. def run_gui(protector_runner=None):
  234. app = ProtectorGUI(protector_runner=protector_runner)
  235. app.mainloop()
  236. if __name__ == "__main__":
  237. run_gui()