如何处理带有集群的Socket.IO房间?

时间:2016-10-16 19:43:18

标签: javascript node.js websocket socket.io node-cluster

我有一个使用群集的服务器,并使用socke.IO工作我使用粘性会话,但我的房间有问题(我不知道我的方式是否是最好的选择) :群集正在实例化流程,每个流程都有特定数量的会议室。

  • 服务器
    • 流程1
      • Room1
      • 室2
      • Room N
    • 流程2
      • Room1
      • 室2
      • Room N

我将一些用户连接到房间(只有一个进程)的方法是使用路由,用户访问页面,当他尝试与Socket.io建立连接时,我检查URL并用我将他插入一个房间的信息。

我的问题是使用集群实现此服务器我无法将用户插入特定的房间,因为有些房间只存在于特定进程中,而粘性会话将他置于另一个进程中。如何将用户置于另一个进程中的房间?此外,使用只能查看他在服务器中的进程路由,我想显示页面中的每个房间。

我已经阅读过有关Redis-Adapter的内容,但我没有在使用Socket.io + Cluster(Sticky-session + redis-adapter)+ rooms的github上找到解决方案。

按照我的代码分享我的所作所为:

//Cluster.Master with simplified Code
if (cluster.isMaster) {

   var workers = [];
   // Spawn workers.
   for (var i = 0; i < num_processes; i++) {
      spawn(i);
   }

   // Create the outside facing server listening on our port.
   var server = net.createServer({
        pauseOnConnect: true
   }, function(connection) {
        // We received a connection and need to pass it to the appropriate
        // worker. Get the worker for this connection's source IP and pass
       // it the connection.
       var worker = workers[worker_index(connection.remoteAddress, num_processes)];
       worker.send('sticky-session:connection', connection);
   }).listen(process.env.PORT);
} else {
     console.log('I am worker #' + cluster.worker.id);
     var app = new express();

     //view engine
     app.set('views', './views');
     app.set('view engine', 'pug');
     //statics
     app.use(express.static(path.join(__dirname, 'public')));
     //rooms
     app.use('/', rooms);
     var server = app.listen(0, 'localhost'),
         io = sio(server);
     io.adapter(sio_redis({ host: 'localhost', port: 6379 }));

    //This File has the socket events (socket.on('messageX', function(){}))
    // And there I am 
    var realtime = require('./realtime/socketIOEvents.js')(io);

    // Listen to messages sent from the master. Ignore everything else.
    process.on('message', function(message, connection) {
    if (message !== 'sticky-session:connection') {
        return;
    }
   // Emulate a connection event on the server by emitting the
   // event with the connection the master sent us.
   server.emit('connection', connection);
   connection.resume();
});
}

3 个答案:

答案 0 :(得分:2)

socketio-redis是正确的做法。每个sserver /进程都会侦听redis队列中的主题。要简单了解socketio-redis,只需将事件发布到集群中的每个其他服务器/进程。 因此,就房间而言,它们只是对一组有兴趣听取房间信息的插座的抽象。

即使套接字分发到不同的服务器/进程,它们也可以是同一个房间的一部分。当每条消息都进来时,每个服务器都知道它并传递给所需的套接字。

就正确性而言,您的架构也是正确的,因为您的代理也决定有选择地转发消息,但它会在消息的生命周期中增加跳数。你真的不需要这个代理来处理套接字路由。

答案 1 :(得分:1)

以下是基于加入的房间工作的示例代码。

  

http://www.html5gamedevs.com/topic/12321-create-a-cluster-server-with-nodejs-and-socketio/

如果您对此帖有疑问,请与我联系。

答案 2 :(得分:1)

经过多次尝试后,我在工作人员的顶部使用代理服务器,并在另一台服务器(Redis)中注册了每个工作人员。当我使用房间的URL连接到服务器时,我的代理服务器会检测我尝试连接的工作人员,并将我的Web套接字连接路由到正确的工作人员。

                      Proxy
           /            |          \
worker(5rooms)   worker(5rooms)  worker(5rooms)
           \            |          /
                      Redis

这解决了我的问题,但它没有使用 socket.io-redis 。我们如何用socket.io-redis解决这个问题?在redis中添加每个用户,套接字和变量,只是在worker中处理是一种有效的方法吗?