170 lines
5.8 KiB
Python
170 lines
5.8 KiB
Python
"""WebSocket 日志推送模块"""
|
||
|
||
from fastapi import APIRouter, WebSocket, WebSocketDisconnect, Query
|
||
from typing import Set, Optional
|
||
import json
|
||
from pathlib import Path
|
||
from src.common.logger import get_logger
|
||
from src.webui.core import get_token_manager
|
||
from src.webui.routers.websocket.auth import verify_ws_token
|
||
|
||
logger = get_logger("webui.logs_ws")
|
||
router = APIRouter()
|
||
|
||
# 全局 WebSocket 连接池
|
||
active_connections: Set[WebSocket] = set()
|
||
|
||
|
||
def load_recent_logs(limit: int = 100) -> list[dict]:
|
||
"""从日志文件中加载最近的日志
|
||
|
||
Args:
|
||
limit: 返回的最大日志条数
|
||
|
||
Returns:
|
||
日志列表
|
||
"""
|
||
logs = []
|
||
log_dir = Path("logs")
|
||
|
||
if not log_dir.exists():
|
||
return logs
|
||
|
||
# 获取所有日志文件,按修改时间排序
|
||
log_files = sorted(log_dir.glob("app_*.log.jsonl"), key=lambda f: f.stat().st_mtime, reverse=True)
|
||
|
||
# 用于生成唯一 ID 的计数器
|
||
log_counter = 0
|
||
|
||
# 从最新的文件开始读取
|
||
for log_file in log_files:
|
||
if len(logs) >= limit:
|
||
break
|
||
|
||
try:
|
||
with open(log_file, "r", encoding="utf-8") as f:
|
||
lines = f.readlines()
|
||
# 从文件末尾开始读取
|
||
for line in reversed(lines):
|
||
if len(logs) >= limit:
|
||
break
|
||
try:
|
||
log_entry = json.loads(line.strip())
|
||
# 转换为前端期望的格式
|
||
# 使用时间戳 + 计数器生成唯一 ID
|
||
timestamp_id = (
|
||
log_entry.get("timestamp", "0").replace("-", "").replace(" ", "").replace(":", "")
|
||
)
|
||
formatted_log = {
|
||
"id": f"{timestamp_id}_{log_counter}",
|
||
"timestamp": log_entry.get("timestamp", ""),
|
||
"level": log_entry.get("level", "INFO").upper(),
|
||
"module": log_entry.get("logger_name", ""),
|
||
"message": log_entry.get("event", ""),
|
||
}
|
||
logs.append(formatted_log)
|
||
log_counter += 1
|
||
except (json.JSONDecodeError, KeyError):
|
||
continue
|
||
except Exception as e:
|
||
logger.error(f"读取日志文件失败 {log_file}: {e}")
|
||
continue
|
||
|
||
# 反转列表,使其按时间顺序排列(旧到新)
|
||
return list(reversed(logs))
|
||
|
||
|
||
@router.websocket("/ws/logs")
|
||
async def websocket_logs(websocket: WebSocket, token: Optional[str] = Query(None)):
|
||
"""WebSocket 日志推送端点
|
||
|
||
客户端连接后会持续接收服务器端的日志消息
|
||
支持三种认证方式(按优先级):
|
||
1. query 参数 token(推荐,通过 /api/webui/ws-token 获取临时 token)
|
||
2. Cookie 中的 maibot_session
|
||
|
||
示例:ws://host/ws/logs?token=xxx
|
||
"""
|
||
is_authenticated = False
|
||
|
||
# 方式 1: 尝试验证临时 WebSocket token(推荐方式)
|
||
if token and verify_ws_token(token):
|
||
is_authenticated = True
|
||
logger.debug("WebSocket 使用临时 token 认证成功")
|
||
|
||
# 方式 2: 尝试从 Cookie 获取 session token
|
||
if not is_authenticated:
|
||
cookie_token = websocket.cookies.get("maibot_session")
|
||
if cookie_token:
|
||
token_manager = get_token_manager()
|
||
if token_manager.verify_token(cookie_token):
|
||
is_authenticated = True
|
||
logger.debug("WebSocket 使用 Cookie 认证成功")
|
||
|
||
if not is_authenticated:
|
||
logger.warning("WebSocket 连接被拒绝:认证失败")
|
||
await websocket.close(code=4001, reason="认证失败,请重新登录")
|
||
return
|
||
|
||
await websocket.accept()
|
||
active_connections.add(websocket)
|
||
logger.info(f"📡 WebSocket 客户端已连接(已认证),当前连接数: {len(active_connections)}")
|
||
|
||
# 连接建立后,立即发送历史日志
|
||
try:
|
||
recent_logs = load_recent_logs(limit=100)
|
||
logger.info(f"发送 {len(recent_logs)} 条历史日志到客户端")
|
||
|
||
for log_entry in recent_logs:
|
||
await websocket.send_text(json.dumps(log_entry, ensure_ascii=False))
|
||
except Exception as e:
|
||
logger.error(f"发送历史日志失败: {e}")
|
||
|
||
try:
|
||
# 保持连接,等待客户端消息或断开
|
||
while True:
|
||
# 接收客户端消息(用于心跳或控制指令)
|
||
data = await websocket.receive_text()
|
||
|
||
# 可以处理客户端的控制消息,例如:
|
||
# - "ping" -> 心跳检测
|
||
# - {"filter": "ERROR"} -> 设置日志级别过滤
|
||
if data == "ping":
|
||
await websocket.send_text("pong")
|
||
|
||
except WebSocketDisconnect:
|
||
active_connections.discard(websocket)
|
||
logger.info(f"📡 WebSocket 客户端已断开,当前连接数: {len(active_connections)}")
|
||
except Exception as e:
|
||
logger.error(f"❌ WebSocket 错误: {e}")
|
||
active_connections.discard(websocket)
|
||
|
||
|
||
async def broadcast_log(log_data: dict):
|
||
"""广播日志到所有连接的 WebSocket 客户端
|
||
|
||
Args:
|
||
log_data: 日志数据字典
|
||
"""
|
||
if not active_connections:
|
||
return
|
||
|
||
# 格式化为 JSON
|
||
message = json.dumps(log_data, ensure_ascii=False)
|
||
|
||
# 记录需要断开的连接
|
||
disconnected = set()
|
||
|
||
# 广播到所有客户端
|
||
for connection in active_connections:
|
||
try:
|
||
await connection.send_text(message)
|
||
except Exception:
|
||
# 发送失败,标记为断开
|
||
disconnected.add(connection)
|
||
|
||
# 清理断开的连接
|
||
if disconnected:
|
||
active_connections.difference_update(disconnected)
|
||
logger.debug(f"清理了 {len(disconnected)} 个断开的 WebSocket 连接")
|