webrtc信令服务器实现

WebRTC 1v1 视频通话技术方案(Node.js + WebSocket)


一、整体架构设计

1.1 系统架构图

  1. 浏览器A ─────┐
  2. (SDP / ICE 交换)
  3. 浏览器B ─────┼──── WebSocket ─── 信令服务器 (Node.js)
  4. 媒体流 (P2P / TURN 中继)

1.2 组件职责

组件 职责
前端浏览器 建立 RTCPeerConnection,采集音视频
信令服务器 交换 offer / answer / ice
TURN 服务器 NAT 穿透失败时中继媒体流
STUN 获取公网候选地址

二、信令服务器技术选型

技术 原因
Express 快速搭建 HTTP 服务
ws 轻量级 WebSocket 实现
原生 Map 管理房间

信令服务器不参与媒体传输,只负责:

  • 房间管理
  • 消息转发

三、核心代码结构解析

3.0通信流程

image-20260213085510995


3.1 静态资源加载

  1. app.use(express.static(path.join(__dirname, "..", "web")));

作用


3.2 WebSocket 服务器创建

  1. const wss = new WebSocketServer({ server });

说明:

  • 复用 HTTP 服务器
  • WebSocket 与 HTTP 共享 3000 端口

四、房间模型设计

  1. const rooms = new Map();

结构:

  1. Map<roomId, Set<ws>>

示例:

  1. rooms = {
  2. "1001" => Set(wsA, wsB)
  3. }

设计思想:

  • 每个房间最多 2 人
  • 每个 ws 保存自己所属 roomId

五、信令流程设计

5.1 加入房间流程

客户端发送:

  1. {
  2. "type": "join",
  3. "roomId": "1001"
  4. }

服务器逻辑:

  1. 检查 roomId
  2. 从旧房间移除
  3. 如果人数 >= 2 返回 full
  4. 加入房间
  5. 返回 joined
  6. 如果人数 == 2,发送 ready

5.2 ready 阶段

当房间人数 = 2 时:

  1. send(peer, { type: "ready", isInitiator });

设计:

  • 后加入者作为 initiator
  • initiator 负责创建 offer

5.3 WebRTC 协商流程

时序图

  1. Initiator Receiver
  2. | |
  3. | ---- offer ----------> |
  4. | |
  5. | <--- answer -----------|
  6. | |
  7. | <--- ICE --------------|
  8. | ---- ICE ------------> |

5.4 信令转发逻辑

  1. if (type === "offer" || type === "answer" || type === "ice") {
  2. relayToOthers(roomId, ws, msg);
  3. }

服务器不解析 SDP,不参与媒体。

它只是“转发器”。


六、核心函数解析


6.1 send()

  1. function send(ws, data) {
  2. if (ws.readyState === ws.OPEN)
  3. ws.send(JSON.stringify(data));
  4. }

说明:

  • 统一发送 JSON
  • 防止向已关闭连接发送数据

6.2 relayToOthers()

  1. function relayToOthers(roomId, sender, data)

功能:

  • 向房间内除自己外的人广播
  • 实现一对一信令交换

6.3 removeFromRoom()

负责:

  • 用户离开
  • 通知房间剩余成员
  • 清理空房间

防止内存泄漏。


七、连接生命周期管理

  1. ws.on("close", () => removeFromRoom(ws));
  2. ws.on("error", () => removeFromRoom(ws));

说明:

  • 防止异常断开导致房间脏数据
  • 保证 rooms 数据一致性

完整源码

  1. import express from "express";
  2. import http from "http";
  3. import { WebSocketServer } from "ws";
  4. import path from "path";
  5. import { fileURLToPath } from "url";
  6. const __filename = fileURLToPath(import.meta.url);
  7. const __dirname = path.dirname(__filename);
  8. const app = express();
  9. // 把 web 目录设置成:server 的上一级 + /web
  10. app.use(express.static(path.join(__dirname, "..", "web")));
  11. const server = http.createServer(app);
  12. const wss = new WebSocketServer({ server });
  13. /**
  14. * rooms: Map<roomId, Set<ws>>
  15. */
  16. const rooms = new Map();
  17. /** 给某个 ws 发送 JSON */
  18. function send(ws, data) {
  19. if (ws.readyState === ws.OPEN) ws.send(JSON.stringify(data));
  20. }
  21. /** 房间内给除了自己之外的其他人转发 */
  22. function relayToOthers(roomId, sender, data) {
  23. const set = rooms.get(roomId);
  24. if (!set) return;
  25. for (const peer of set) {
  26. if (peer !== sender) send(peer, data);
  27. }
  28. }
  29. function removeFromRoom(ws) {
  30. const roomId = ws.roomId;
  31. if (!roomId) return;
  32. const set = rooms.get(roomId);
  33. if (!set) return;
  34. set.delete(ws);
  35. // 通知剩余的人:对方离开
  36. for (const peer of set) {
  37. send(peer, { type: "peer-left" });
  38. }
  39. if (set.size === 0) rooms.delete(roomId);
  40. ws.roomId = null;
  41. }
  42. wss.on("connection", (ws) => {
  43. ws.roomId = null;
  44. ws.on("message", (buf) => {
  45. let msg;
  46. try {
  47. msg = JSON.parse(buf.toString());
  48. } catch {
  49. return;
  50. }
  51. const { type } = msg;
  52. if (type === "join") {
  53. const roomId = String(msg.roomId || "").trim();
  54. if (!roomId) return send(ws, { type: "error", message: "roomId is required" });
  55. // 如果 ws 已在别的房间,先移除
  56. removeFromRoom(ws);
  57. const set = rooms.get(roomId) || new Set();
  58. if (set.size >= 2) {
  59. return send(ws, { type: "full", roomId });
  60. }
  61. set.add(ws);
  62. rooms.set(roomId, set);
  63. ws.roomId = roomId;
  64. send(ws, { type: "joined", roomId, peers: set.size - 1 });
  65. // 房间凑齐 2 人,通知双方开始协商
  66. if (set.size === 2) {
  67. // 约定:后加入的人做 initiator(也可以反过来)
  68. for (const peer of set) {
  69. const isInitiator = peer === ws;
  70. send(peer, { type: "ready", isInitiator });
  71. }
  72. }
  73. return;
  74. }
  75. // 后面的消息都必须在房间里
  76. const roomId = ws.roomId;
  77. if (!roomId) return send(ws, { type: "error", message: "join a room first" });
  78. // 透传 WebRTC 协商消息
  79. if (type === "offer" || type === "answer" || type === "ice") {
  80. relayToOthers(roomId, ws, msg);
  81. return;
  82. }
  83. if (type === "leave") {
  84. removeFromRoom(ws);
  85. send(ws, { type: "left" });
  86. return;
  87. }
  88. });
  89. ws.on("close", () => removeFromRoom(ws));
  90. ws.on("error", () => removeFromRoom(ws));
  91. });
  92. const PORT = process.env.PORT || 3000;
  93. server.listen(PORT, "0.0.0.0", () => {
  94. console.log(`Signaling+Web server running: http://localhost:${PORT}`);
  95. });

八、为什么信令服务器不处理媒体?

因为:

WebRTC 是 P2P 协议。

媒体路径:

  1. 浏览器A ←→ 浏览器B

不是:

  1. 浏览器A 服务器 浏览器B

除非使用 SFU。


九、当前版本限制

项目 当前实现
房间人数 最多 2 人
认证
房间权限
重连机制
多人视频 不支持

十、如何扩展为多人房间(技术方向)

当前结构:

  1. Map<roomId, Set<ws>>

升级方案:

  1. 为每个 ws 分配唯一 peerId
  2. 信令改为定向发送
  3. 前端维护:
  1. Map<peerId, RTCPeerConnection>

每加入一个人:

  • 为其创建一个新的 PeerConnection
  • 动态创建 video 元素

这叫:

Mesh 架构


十一、生产环境建议

1️⃣ 使用 HTTPS + WSS

WebRTC 在公网通常必须 HTTPS。

2️⃣ TURN 使用动态签名

不要写死:

  1. user=webrtc:password

应改为:

  1. use-auth-secret
  2. static-auth-secret=xxx

防止带宽被盗用。

3️⃣ 加入房间认证

目前任何人知道房间号即可进入。

应加入:

  • token
  • 用户身份

热门评论

热门文章

  1. vscode搭建windows C++开发环境

    喜欢(596) 浏览(102638)
  2. 使用hexo搭建个人博客

    喜欢(533) 浏览(14716)
  3. Linux环境搭建和编码

    喜欢(594) 浏览(16370)
  4. MarkDown在线编辑器

    喜欢(514) 浏览(16776)
  5. 聊天项目(28) 分布式服务通知好友申请

    喜欢(507) 浏览(7564)

最新评论

  1. 利用指针和容器实现文本查询 越今朝:应该添加一个过滤功能以解决部分单词无法被查询的问题: eg: "I am a teacher."中的teacher无法被查询,因为在示例代码中teacher.被解释为一个单词从而忽略了teacher本身。
  2. 无锁并发队列 TenThousandOne:_head  和 _tail  替换为原子变量。那里pop的逻辑,val = _data[h] 可以移到循环外面吗
  3. 解决博客回复区被脚本注入的问题 secondtonone1:走到现在我忽然明白一个道理,无论工作也好生活也罢,最重要的是开心,即使一份安稳的工作不能给我带来事业上的积累也要合理的舍弃,所以我还是想去做喜欢的方向。
  4. 处理网络粘包问题 zyouth: //消息的长度小于头部规定的长度,说明数据未收全,则先将部分消息放到接收节点里 if (bytes_transferred < data_len) { memcpy(_recv_msg_node->_data + _recv_msg_node->_cur_len, _data + copy_len, bytes_transferred); _recv_msg_node->_cur_len += bytes_transferred; ::memset(_data, 0, MAX_LENGTH); _socket.async_read_some(boost::asio::buffer(_data, MAX_LENGTH), std::bind(&CSession::HandleRead, this, std::placeholders::_1, std::placeholders::_2, shared_self)); //头部处理完成 _b_head_parse = true; return; } 把_b_head_parse = true;放在_socket.async_read_some前面是不是更好
  5. C++ 线程池原理和实现 mzx2023:两种方法解决,一种是改排序算法,就是当线程耗尽的时候,使用普通递归,另一种是当在线程池commit的时候,判断线程是否耗尽,耗尽的话就直接当前线程执行task

个人公众号

个人微信