Warning: file_get_contents(/data/phpspider/zhask/data//catemap/9/javascript/461.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181

Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/node.js/40.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Javascript 如何使用群集处理Socket.IO文件室?_Javascript_Node.js_Websocket_Socket.io_Node Cluster - Fatal编程技术网

Javascript 如何使用群集处理Socket.IO文件室?

Javascript 如何使用群集处理Socket.IO文件室?,javascript,node.js,websocket,socket.io,node-cluster,Javascript,Node.js,Websocket,Socket.io,Node Cluster,我有一台服务器在使用集群,为了使它能够使用socke.IO,我使用的是sticky session,但我的文件室有一个问题(我不知道我这样做是否是最好的选择):集群正在实例化进程,每个进程都有特定数量的文件室 服务器 过程1 1号房间 房间2 N房间 过程2 1号房间 房间2 N房间 我将一些用户连接到文件室的方法(仅使用一个进程)是使用路由,用户在其中访问一个页面,当他尝试与Socket.io建立连接时,我检查URL,并根据该信息将他插入文件室 我的问题是使用群集实现此服务器。

我有一台服务器在使用集群,为了使它能够使用socke.IO,我使用的是sticky session,但我的文件室有一个问题(我不知道我这样做是否是最好的选择):集群正在实例化进程,每个进程都有特定数量的文件室

  • 服务器
    • 过程1
      • 1号房间
      • 房间2
      • N房间
    • 过程2
      • 1号房间
      • 房间2
      • N房间
我将一些用户连接到文件室的方法(仅使用一个进程)是使用路由,用户在其中访问一个页面,当他尝试与Socket.io建立连接时,我检查URL,并根据该信息将他插入文件室

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

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

按照我的代码分享我所做的工作:

//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();
});
}
//带有简化代码的Cluster.Master
if(cluster.isMaster){
var工人=[];
//产卵工人。
对于(var i=0;i
以下是示例代码,该代码将根据与的房间连接而起作用


如果你在这篇文章中有问题,请告诉我

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

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

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

socketio redis是正确的选择。每个sserver/进程侦听redis队列中的一个主题。简而言之,socketio redis只是将事件发布到集群中的每个其他服务器/进程。 因此,就房间而言,它们只是一组对房间中的消息感兴趣的套接字的抽象

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


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

redis connector for socket.io就是为了解决这个问题而设计的。你有更好的例子来说明如何使用它吗?SocketIO和SocketIO redis的文档非常糟糕:/Links 3-5:我已经看到了文档,它对我没有帮助,因为redis示例只是在服务器之间共享事件。。。由于我有一个房间管理,我是否应该在每台服务器中复制逻辑(用户加入房间X、userleft、用户发送消息X、Y、Z)?如果您使用socket.io房间,那么您可以添加到房间并向房间发送消息,socket.io/redis适配器会自动为您处理多服务器事务。它在redis中存储每个用户是哪个服务器,并在redis中保留房间列表,因此当您想要向房间广播时,它会转到redis,获取房间中的用户列表,然后告诉服务器每个用户发送给他们。我想你的问题已经解决了。如果不是,那么您没有描述为什么内置解决方案不是您想要或需要的。