Reversed the wrong conclusion in 46bba0d. The vanilla L4D2 logaddress
UDP path is NOT broken — proved by retesting with destination
172.30.0.5:28000 (wireguard IP) which yielded 8 properly framed HL
Log Standard packets in 12s including real game events.
Root cause: the Source engine silently drops logaddress destinations
in 127.0.0.0/8. Registration succeeds and the cvar API reports
"logging to: udp" but sendto is never called for loopback. Every
other L4D2 stats deployment (multiple production HLstatsX:CE
instances) puts the collector on a separate host or interface IP
and never hits this.
Defaults: LOG_LISTENER_BIND=0.0.0.0:28000 (accept on any interface);
LOG_LISTENER_ADDR="" (production must set via web.env to the host's
non-loopback IP). Empty default = safe no-op for dev. The kernel's
same-host routing optimization keeps the traffic on lo internally
but the packet's destination IP must not literally be in 127/8.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
65 lines
3.2 KiB
Python
65 lines
3.2 KiB
Python
import os
|
|
|
|
|
|
DEFAULT_CONFIG: dict[str, object] = {
|
|
"SECRET_KEY": None,
|
|
"DATABASE_URL": "sqlite:///l4d2web.db",
|
|
"STATUS_REFRESH_SECONDS": 8,
|
|
"JOB_WORKER_THREADS": 4,
|
|
"JOB_WORKER_ENABLED": True,
|
|
"JOB_WORKER_POLL_SECONDS": 1,
|
|
"STATE_POLLER_INTERVAL_SECONDS": 30,
|
|
"JOB_LOG_REPLAY_LIMIT": 2000,
|
|
"JOB_LOG_LINE_MAX_CHARS": 4096,
|
|
"PORT_RANGE_START": 27015,
|
|
"PORT_RANGE_END": 27115,
|
|
"LIVE_STATE_POLL_SECONDS": 5,
|
|
"LIVE_STATE_QUERY_TIMEOUT_SECONDS": 2.0,
|
|
"LIVE_STATE_STALE_SECONDS": 30,
|
|
"LIVE_STATE_HISTORY_DAYS": 30,
|
|
"LIVE_STATE_RETENTION_EVERY_TICKS": 60,
|
|
"STUCK_SESSION_SECONDS": 60,
|
|
"STEAM_PROFILE_TTL_SECONDS": 86400,
|
|
"STEAM_WEB_API_KEY": "",
|
|
"LOG_LISTENER_ENABLED": True,
|
|
"LOG_LISTENER_BIND": "0.0.0.0:28000",
|
|
# Empty by default — production MUST set this to a non-loopback IP via
|
|
# web.env. Source silently drops logaddress destinations in 127.0.0.0/8,
|
|
# so 127.0.0.1 here would register but never receive packets. When
|
|
# empty, the cfg injector skips emitting `log on` / `logaddress_add`,
|
|
# which is the safer no-op for dev environments without a srcds host.
|
|
"LOG_LISTENER_ADDR": "",
|
|
"LOG_CAPTURE_DIR": "/var/lib/left4me/captures",
|
|
}
|
|
|
|
|
|
def _bool_from_env(raw: str) -> bool:
|
|
return raw.lower() not in {"0", "false", "no"}
|
|
|
|
|
|
def load_config() -> dict[str, object]:
|
|
return {
|
|
"SECRET_KEY": os.getenv("SECRET_KEY"),
|
|
"DATABASE_URL": os.getenv("DATABASE_URL", "sqlite:///l4d2web.db"),
|
|
"STATUS_REFRESH_SECONDS": int(os.getenv("STATUS_REFRESH_SECONDS", "8")),
|
|
"JOB_WORKER_THREADS": int(os.getenv("JOB_WORKER_THREADS", "4")),
|
|
"JOB_WORKER_ENABLED": _bool_from_env(os.getenv("JOB_WORKER_ENABLED", "true")),
|
|
"JOB_WORKER_POLL_SECONDS": float(os.getenv("JOB_WORKER_POLL_SECONDS", "1")),
|
|
"STATE_POLLER_INTERVAL_SECONDS": float(os.getenv("STATE_POLLER_INTERVAL_SECONDS", "30")),
|
|
"JOB_LOG_REPLAY_LIMIT": int(os.getenv("JOB_LOG_REPLAY_LIMIT", "2000")),
|
|
"JOB_LOG_LINE_MAX_CHARS": int(os.getenv("JOB_LOG_LINE_MAX_CHARS", "4096")),
|
|
"PORT_RANGE_START": int(os.getenv("LEFT4ME_PORT_RANGE_START", "27015")),
|
|
"PORT_RANGE_END": int(os.getenv("LEFT4ME_PORT_RANGE_END", "27115")),
|
|
"LIVE_STATE_POLL_SECONDS": float(os.getenv("LIVE_STATE_POLL_SECONDS", "5")),
|
|
"LIVE_STATE_QUERY_TIMEOUT_SECONDS": float(os.getenv("LIVE_STATE_QUERY_TIMEOUT_SECONDS", "2.0")),
|
|
"LIVE_STATE_STALE_SECONDS": int(os.getenv("LIVE_STATE_STALE_SECONDS", "30")),
|
|
"LIVE_STATE_HISTORY_DAYS": int(os.getenv("LIVE_STATE_HISTORY_DAYS", "30")),
|
|
"LIVE_STATE_RETENTION_EVERY_TICKS": int(os.getenv("LIVE_STATE_RETENTION_EVERY_TICKS", "60")),
|
|
"STUCK_SESSION_SECONDS": int(os.getenv("STUCK_SESSION_SECONDS", "60")),
|
|
"STEAM_PROFILE_TTL_SECONDS": int(os.getenv("STEAM_PROFILE_TTL_SECONDS", "86400")),
|
|
"STEAM_WEB_API_KEY": os.getenv("STEAM_WEB_API_KEY", ""),
|
|
"LOG_LISTENER_ENABLED": _bool_from_env(os.getenv("LOG_LISTENER_ENABLED", "true")),
|
|
"LOG_LISTENER_BIND": os.getenv("LOG_LISTENER_BIND", "0.0.0.0:28000"),
|
|
"LOG_LISTENER_ADDR": os.getenv("LOG_LISTENER_ADDR", ""),
|
|
"LOG_CAPTURE_DIR": os.getenv("LOG_CAPTURE_DIR", "/var/lib/left4me/captures"),
|
|
}
|