14 KiB
14 KiB
WebSocket 服务器
**本文引用的文件** - [src/index.ts](file://src/index.ts) - [src/server.ts](file://src/server.ts) - [src/websocket.ts](file://src/websocket.ts) - [src/class/websockethandler.ts](file://src/class/websockethandler.ts) - [src/class/offer.ts](file://src/class/offer.ts) - [src/class/answer.ts](file://src/class/answer.ts) - [src/class/candidate.ts](file://src/class/candidate.ts) - [src/log.ts](file://src/log.ts) - [src/signaling.ts](file://src/signaling.ts) - [src/swagger.ts](file://src/swagger.ts) - [client/src/signaling.js](file://client/src/signaling.js) - [package.json](file://package.json) - [test/websockethandler.test.ts](file://test/websockethandler.test.ts)目录
简介
本技术文档围绕 WebSocket 服务器模块进行深入解析,重点覆盖以下方面:
- WebSocket 服务器的创建与配置(含 HTTPS/WSS、日志与启动流程)
- 连接生命周期管理(连接建立、断开、广播、消息路由)
- WSSignaling 类的职责与实现要点(事件监听、消息分发、心跳与超时)
- WebSocketHandler 的核心能力(连接组模型、1对多/多对多路由、广播机制)
- 连接调试与监控方法(日志级别、Swagger 文档、心跳检测)
- 连接池管理、心跳检测与断线重连策略
项目结构
后端采用 Express 提供 HTTP/HTTPS 服务与静态资源,WebSocket 信令在独立模块中运行;同时保留 HTTP 轮询信令作为备选方案。
graph TB
subgraph "服务端进程"
A["Express 应用<br/>src/server.ts"]
B["HTTP(S) 服务器<br/>src/index.ts"]
C["WebSocket 信令服务器<br/>src/websocket.ts"]
D["WebSocket 处理器<br/>src/class/websockethandler.ts"]
E["HTTP 信令路由<br/>src/signaling.ts"]
F["Swagger 文档<br/>src/swagger.ts"]
end
subgraph "客户端"
G["浏览器/前端页面<br/>client/public"]
H["信令客户端封装<br/>client/src/signaling.js"]
end
H --> |"WebSocket"| C
H --> |"HTTP 轮询"| E
B --> |"创建/绑定"| C
A --> |"挂载路由/静态资源"| G
A --> |"注册 HTTP 信令路由"| E
A --> |"初始化 Swagger"| F
图表来源
- src/index.ts:52-91
- src/server.ts:14-42
- src/websocket.ts:15-116
- src/signaling.ts:4-24
- src/swagger.ts:16-65
章节来源
- src/index.ts:13-109
- src/server.ts:14-90
- src/websocket.ts:6-118
- src/signaling.ts:1-25
- src/swagger.ts:16-65
核心组件
- WSSignaling:WebSocket 信令服务器入口,负责监听连接、消息解析与分发至 WebSocketHandler。
- WebSocketHandler:核心业务逻辑,维护连接组、1对多/多对多路由、广播、心跳与断线清理。
- 数据模型:Offer、Answer、Candidate 用于封装信令数据。
- 日志系统:统一日志级别控制与输出格式。
- HTTP 信令路由:保留 HTTP 轮询信令通道,便于兼容与调试。
- Swagger 文档:自动生成 API 文档,支持会话认证与信令接口说明。
章节来源
- src/websocket.ts:6-118
- src/class/websockethandler.ts:1-479
- src/class/offer.ts:1-11
- src/class/answer.ts:1-8
- src/class/candidate.ts:1-12
- src/log.ts:1-51
- src/signaling.ts:1-25
- src/swagger.ts:16-65
架构总览
WebSocket 服务器在 HTTP(S) 之上提供实时信令通道,客户端可通过 WebSocket 或 HTTP 轮询两种方式接入。WebSocket 信令以“连接组”为核心组织多端通信,支持广播与定向转发。
sequenceDiagram
participant Client as "客户端<br/>client/src/signaling.js"
participant WS as "WSSignaling<br/>src/websocket.ts"
participant Handler as "WebSocketHandler<br/>src/class/websockethandler.ts"
Client->>WS : "WebSocket 连接建立"
WS->>Handler : "add(ws)"
WS-->>Client : "connect 消息role/polite/participantId"
Client->>WS : "发送信令消息offer/answer/candidate/on-message/broadcast"
WS->>Handler : "根据消息类型分发"
Handler-->>Client : "转发/广播结果含 participantId"
Client-->>WS : "断开连接/心跳 ping/pong"
WS->>Handler : "remove(ws)/心跳检测"
图表来源
- src/websocket.ts:27-116
- src/class/websockethandler.ts:72-137
- src/class/websockethandler.ts:404-430
- client/src/signaling.js:152-292
详细组件分析
WSSignaling 类
职责与行为:
- 接收 HTTP 服务器实例,创建 ws.Server 并绑定到同一端口。
- 注册 connection 事件:为新连接调用处理器 add,并在关闭时 remove。
- 注册 message 事件:解析 JSON 消息,按 type 分发到对应处理器(connect/disconnect/offer/answer/candidate/ping/pong/broadcast/on-message/call-request)。
- 内置 ping/pong 心跳:收到 ping 回复 pong,收到 pong 更新 lastActivity;注释掉的心跳定时器可在需要时启用。
实现要点:
- 模式切换:通过 reset(mode) 设置 isPrivate,影响连接组与路由行为。
- 消息格式:支持 participantId 字段以区分多参与者场景。
- 日志:统一通过 log 输出接收到的消息与关键事件。
章节来源
WebSocketHandler 核心功能
- 连接组模型:Map<string, ConnectionGroup>,每个连接组包含 host 与多个 participants。
- 连接管理:add/remove 维护 clients 映射与连接组;onConnect/onDisconnect 实现 1对多/多对多角色分配。
- 信令路由:
- offer/answer/candidate:host 与 participants 之间双向转发;私有模式下支持按 participantId 精确路由。
- broadcast:支持全局广播与按连接组广播。
- on-message:支持聊天消息等扩展消息的组内转发。
- 心跳与超时:提供心跳初始化/清理函数,注释掉的定时器默认不启用;可按需开启。
- 辅助能力:获取所有连接组 ID、判断是否 host、组内广播工具。
classDiagram
class WSSignaling {
+server
+wss
+constructor(server, mode)
}
class WebSocketHandler {
+reset(mode)
+add(ws)
+remove(ws)
+onConnect(ws, connectionId)
+onDisconnect(ws, connectionId)
+onOffer(ws, message)
+onAnswer(ws, message)
+onCandidate(ws, message)
+onBroadcast(ws, message)
+onMessage(ws, message)
+AddHeartbeat(ws, connectionId)
+RemoveHeartbeat(ws)
+isHost(ws, connectionId)
+broadcastToGroup(connectionId, senderWs, message)
}
WSSignaling --> WebSocketHandler : "消息分发"
图表来源
章节来源
- src/class/websockethandler.ts:139-206
- src/class/websockethandler.ts:208-338
- src/class/websockethandler.ts:340-402
- src/class/websockethandler.ts:404-430
数据模型
- Offer:封装 SDP、时间戳与是否“polite”标记。
- Answer:封装 SDP 与时间戳。
- Candidate:封装 ICE 候选项与 SDP 元信息及时间戳。
章节来源
启动与配置
- 命令行参数:端口、HTTPS 开关与证书、信令类型(websocket/http)、通信模式(public/private)、日志级别。
- HTTPS:当 secure=true 时使用 server.key/server.cert 创建 https.Server。
- 信令类型:type=websocket 时启动 WSSignaling;否则提示并回退为 websocket。
- 日志:统一通过 log 模块输出启动信息与运行日志。
章节来源
HTTP 信令与 Swagger
- HTTP 信令路由:提供会话管理与信令 CRUD 接口,支持会话认证(session-id)。
- Swagger:自动扫描 httphandler.ts 与 signaling.ts,生成 API 文档,支持会话认证与信令接口说明。
章节来源
依赖关系分析
- 运行时依赖:ws(WebSocket)、express(HTTP)、morgan(日志)、swagger-ui-express(文档)、uuid(会话 ID)。
- 构建与测试:TypeScript、Jest、ESLint、ts-jest 等。
- 客户端:浏览器通过 WebSocket 或 HTTP 轮询与服务端交互。
graph LR
P["package.json 依赖声明"] --> WS["ws"]
P --> EXP["express"]
P --> MORGAN["morgan"]
P --> SWG["swagger-ui-express"]
P --> UUID["uuid"]
IDX["src/index.ts"] --> SRV["src/server.ts"]
IDX --> WSC["src/websocket.ts"]
WSC --> WSH["src/class/websockethandler.ts"]
SRV --> SIG["src/signaling.ts"]
SRV --> SWG
图表来源
- package.json:14-46
- src/index.ts:7-11
- src/server.ts:5-9
- src/websocket.ts:1-4
- src/class/websockethandler.ts:5-8
- src/signaling.ts:1-3
- src/swagger.ts:5-9
章节来源
- package.json:14-60
- src/index.ts:7-11
- src/server.ts:5-9
- src/websocket.ts:1-4
- src/class/websockethandler.ts:5-8
- src/signaling.ts:1-3
- src/swagger.ts:5-9
性能考量
- 连接组规模:每组 participants 使用 Set 存储,查找与删除均为 O(1),适合中小规模并发。
- 广播策略:组内广播遍历 participants,建议限制单组成员数量或引入分区策略。
- 心跳检测:当前注释掉定时器,避免不必要的 CPU 占用;如启用需合理设置周期与超时阈值。
- 日志级别:生产环境建议提升日志级别,减少高频日志输出对性能的影响。
- HTTPS 证书:WSS 需要正确配置证书与密钥,确保握手与传输安全。
故障排查指南
- 启动与网络
- 确认端口占用与防火墙放行。
- HTTPS 模式检查 server.key 与 server.cert 是否存在且可读。
- WebSocket 连接
- 检查客户端 WebSocket URL(ws/wss)与主机一致。
- 观察浏览器开发者工具 Network 面板中的 WebSocket 握手与帧。
- 信令消息
- 使用 Swagger 文档校验 HTTP 信令接口(PUT/GET/POST/DELETE)参数与会话 ID。
- 对照客户端 signaling.js 的消息构造与事件派发。
- 日志定位
- 调整日志级别(info/warn/error/log),关注连接建立、断开、广播与路由关键节点。
- 自动化测试
- 参考单元测试用例,验证 public/private 模式下的连接、offer/answer/candidate 转发与广播行为。
章节来源
- src/index.ts:55-88
- src/log.ts:11-51
- src/swagger.ts:16-65
- client/src/signaling.js:152-292
- test/websockethandler.test.ts:1-191
结论
该 WebSocket 服务器模块以简洁清晰的职责划分实现了 WebRTC 信令的核心需求:支持 WebSocket 与 HTTP 两种信令通道、基于连接组的 1对多/多对多路由、广播与定向转发、以及可扩展的消息类型。通过日志与 Swagger 的配合,具备良好的可观测性与可维护性。建议在高并发场景下进一步优化广播与心跳策略,并完善断线重连与幂等处理。
附录
WebSocket 连接调试与监控
- 浏览器开发者工具:Network 面板查看握手、帧与错误。
- Swagger 文档:访问 /api-docs,使用 session-id 头部进行认证,查看 HTTP 信令接口。
- 日志:通过命令行参数设置日志级别,观察连接、断开、广播与路由事件。
- 心跳:若启用心跳定时器,注意 ping/pong 的周期与超时阈值配置。
章节来源
连接池管理、心跳检测与断线重连
- 连接池管理:clients 与 connectionGroup 维护连接映射与组关系,remove 时清理并广播断开事件。
- 心跳检测:提供 AddHeartbeat/RemoveHeartbeat,当前代码注释掉定时器;可按需启用。
- 断线重连:客户端应实现指数退避与自动重连逻辑;服务端在断开时通知组内成员并清理资源。
章节来源