如何处理 Socket.IO 个带集群的房间?
How can I Handle Socket.IO rooms with cluster?
我有一个与集群一起工作的服务器,并使其与 socke.IO 一起工作我正在使用粘性会话,但我的房间有问题(我不知道我做的方式是否是最佳选择):集群正在实例化进程,每个进程都有特定数量的房间。
- 服务器
- 进程 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();
});
}
这是基于加入房间的示例代码。
http://www.html5gamedevs.com/topic/12321-create-a-cluster-server-with-nodejs-and-socketio/
如果您对此 post 有任何疑问,请告诉我。
经过多次尝试,我在工作人员顶部使用代理服务器并在另一台服务器(Redis)中注册房间的每个工作人员完成了此操作。当我使用房间的 URL 连接到服务器时,我的代理服务器检测到我正在尝试连接哪个工作人员并将我的网络套接字连接路由到正确的工作人员。
Proxy
/ | \
worker(5rooms) worker(5rooms) worker(5rooms)
\ | /
Redis
这解决了我的问题,但它没有使用 socket.io-redis。我们如何用 socket.io-redis 解决这个问题?在 redis 中添加每个用户、套接字和变量以仅在工作人员中进行处理是一种有效的方法吗?
socketio-redis 是正确的做法。每个 sserver/process 监听 redis 队列中的一个主题。简而言之,socketio-redis 只是将事件发布到集群中的每个其他 server/process。
因此,就房间而言,它们只是对一组有兴趣在房间中收听消息的套接字的抽象。
即使插座分布在不同的 servers/processes,它们也可以是同一房间的一部分。当每条消息进来时,每台服务器都知道它并将其传递给所需的套接字。
就正确性而言,您的架构也是正确的,因为您的代理决定也有选择地转发消息,但它增加了消息生命周期中的跳数。你真的不需要这个代理来处理套接字路由。
我有一个与集群一起工作的服务器,并使其与 socke.IO 一起工作我正在使用粘性会话,但我的房间有问题(我不知道我做的方式是否是最佳选择):集群正在实例化进程,每个进程都有特定数量的房间。
- 服务器
- 进程 1
- 房间 1
- 房间2
- N号房间
- 进程 2
- 房间 1
- 房间2
- N号房间
- 进程 1
我将一些用户连接到房间的方法(只有一个过程)是使用路由,用户访问页面的地方,当他试图与 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();
});
}
这是基于加入房间的示例代码。
http://www.html5gamedevs.com/topic/12321-create-a-cluster-server-with-nodejs-and-socketio/
如果您对此 post 有任何疑问,请告诉我。
经过多次尝试,我在工作人员顶部使用代理服务器并在另一台服务器(Redis)中注册房间的每个工作人员完成了此操作。当我使用房间的 URL 连接到服务器时,我的代理服务器检测到我正在尝试连接哪个工作人员并将我的网络套接字连接路由到正确的工作人员。
Proxy
/ | \
worker(5rooms) worker(5rooms) worker(5rooms)
\ | /
Redis
这解决了我的问题,但它没有使用 socket.io-redis。我们如何用 socket.io-redis 解决这个问题?在 redis 中添加每个用户、套接字和变量以仅在工作人员中进行处理是一种有效的方法吗?
socketio-redis 是正确的做法。每个 sserver/process 监听 redis 队列中的一个主题。简而言之,socketio-redis 只是将事件发布到集群中的每个其他 server/process。 因此,就房间而言,它们只是对一组有兴趣在房间中收听消息的套接字的抽象。
即使插座分布在不同的 servers/processes,它们也可以是同一房间的一部分。当每条消息进来时,每台服务器都知道它并将其传递给所需的套接字。
就正确性而言,您的架构也是正确的,因为您的代理决定也有选择地转发消息,但它增加了消息生命周期中的跳数。你真的不需要这个代理来处理套接字路由。