config_tcp.py 3.2 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  1. from __future__ import annotations
  2. import json
  3. from dataclasses import dataclass
  4. from pathlib import Path
  5. from typing import Literal
  6. Strategy = Literal["broadcast", "top2", "top3", "top4", "backup"]
  7. KernelMode = Literal["auto", "20", "24"]
  8. @dataclass
  9. class TcpRelayNode:
  10. name: str
  11. host: str
  12. port: int
  13. token: str
  14. weight: int = 100
  15. @dataclass
  16. class TcpConfig:
  17. relays: list[TcpRelayNode]
  18. strategy: Strategy = "top3"
  19. kernel_mode: KernelMode = "auto"
  20. redundancy: int = 3
  21. tcp_warmup_bytes: int = 1048576
  22. probe_interval: float = 15.0
  23. tcp_loser_grace_ms: int = 1500
  24. direct_open_timeout: float = 10.0
  25. relay_open_timeout: float = 10.0
  26. tcp_connect_happy_eyeballs_delay: float | None = None
  27. ssh_warmup_bytes: int = 4096
  28. ssh_loser_grace_ms: int = 80
  29. direct_ipv6_enabled: bool = True
  30. tcp_failover_idle_ms: int = 1200
  31. relay_reconnect_delay: float = 3.0
  32. relay_reconnect_attempts: int = 5
  33. relay_reconnect_max_delay: float = 30.0
  34. relay_ping_interval: float = 10.0
  35. relay_ping_timeout: float = 25.0
  36. relay_tcp_nodelay: bool = True
  37. direct_redundancy: int = 2
  38. direct_redundancy_v4: int | None = None
  39. direct_redundancy_v6: int | None = None
  40. direct_max_redundancy: int = 3
  41. @classmethod
  42. def from_dict(cls, raw: dict) -> "TcpConfig":
  43. relays = [TcpRelayNode(**item) for item in raw.get("relays", [])]
  44. return cls(
  45. relays=relays,
  46. strategy=raw.get("strategy", "top3"),
  47. kernel_mode=raw.get("kernel_mode", "auto"),
  48. redundancy=raw.get("redundancy", 3),
  49. tcp_warmup_bytes=raw.get("tcp_warmup_bytes", 1048576),
  50. probe_interval=raw.get("probe_interval", 15.0),
  51. tcp_loser_grace_ms=raw.get("tcp_loser_grace_ms", 1500),
  52. direct_open_timeout=raw.get("direct_open_timeout", 10.0),
  53. relay_open_timeout=raw.get("relay_open_timeout", 10.0),
  54. tcp_connect_happy_eyeballs_delay=raw.get("tcp_connect_happy_eyeballs_delay"),
  55. ssh_warmup_bytes=max(0, raw.get("ssh_warmup_bytes", 4096)),
  56. ssh_loser_grace_ms=max(0, raw.get("ssh_loser_grace_ms", 80)),
  57. direct_ipv6_enabled=raw.get("direct_ipv6_enabled", True),
  58. tcp_failover_idle_ms=max(100, raw.get("tcp_failover_idle_ms", 1200)),
  59. relay_reconnect_delay=raw.get("relay_reconnect_delay", 3.0),
  60. relay_reconnect_attempts=max(1, raw.get("relay_reconnect_attempts", 5)),
  61. relay_reconnect_max_delay=max(raw.get("relay_reconnect_delay", 3.0), raw.get("relay_reconnect_max_delay", 30.0)),
  62. relay_ping_interval=max(1.0, raw.get("relay_ping_interval", 10.0)),
  63. relay_ping_timeout=max(1.0, raw.get("relay_ping_timeout", 25.0)),
  64. relay_tcp_nodelay=raw.get("relay_tcp_nodelay", True),
  65. direct_redundancy=max(1, raw.get("direct_redundancy", 2)),
  66. direct_redundancy_v4=raw.get("direct_redundancy_v4"),
  67. direct_redundancy_v6=raw.get("direct_redundancy_v6"),
  68. direct_max_redundancy=max(1, raw.get("direct_max_redundancy", 3)),
  69. )
  70. @classmethod
  71. def load(cls, path: str) -> "TcpConfig":
  72. return cls.from_dict(json.loads(Path(path).read_text()))