[英]Errors going to 2 dynos on Heroku with socket.io / socket.io-redis / rediscloud / node.js
[英]Scale node.js socket.io@1.*.* with cluster and socket.io-redis on Heroku
有谁知道一个很好的解决方案,可以在多个内核上扩展node.js-基于socket.io的应用程序? 我目前正在测试socket.io文档中提供的解决方案,以在多个节点上使用socket.io,但没有取得具体的成功。
我在github上为此创建了一个游乐场: https : //github.com/liviuignat/socket.io-clusters ,它是来自socket.io站点的聊天应用程序的经过修改的副本。 它使用express
, cluster
, socket.io@1.1.0
和socket.io-redis
。
当前,在分支feature/sticky
还有一个使用sticky-session
的实现似乎更好地工作。
最后,该应用程序需要发布到Heroku ,并在多个dyno上进行缩放。
最初,我尝试执行类似的操作-仅为群集节点启动服务器,但始终收到错误消息: 失败:在收到握手响应之前连接已关闭
if (cluster.isMaster) {
for (var i = 0; i < numCPUs; i++) {
cluster.fork();
}
cluster.on('exit', function(worker, code, signal) {
console.log('worker ' + worker.process.pid + ' died');
});
} else {
var server = new Server({
dirName: __dirname,
enableSocket: true
})
.setupApp()
.setupRoutes()
.start();
}
然后,我也尝试为主节点启动服务器:
if (cluster.isMaster) {
var server = new Server({
dirName: __dirname,
enableSocket: true
})
.setupApp()
.setupRoutes()
.start();
for (var i = 0; i < numCPUs; i++) {
cluster.fork();
}
cluster.on('exit', function(worker, code, signal) {
console.log('worker ' + worker.process.pid + ' died');
});
} else {
var server = new Server({
dirName: __dirname,
enableSocket: true
})
.setupApp()
.setupRoutes()
.start();
}
我还在分支feature/sticky
同时使用了sticky-session
和socket.io-redis
进行了尝试,这似乎很成功,但似乎仍然不是一个好的解决方案:
if (cluster.isMaster) {
sticky(function() {
var server = new Server({
dirName: __dirname,
enableSocket: true
})
.setupApp()
.setupRoutes();
return server.http;
}).listen(3000, function() {
console.log('server started on 3000 port');
});
for (var i = 0; i < numCPUs; i++) {
cluster.fork();
}
cluster.on('exit', function(worker, code, signal) {
console.log('worker ' + worker.process.pid + ' died');
});
} else {
sticky(function() {
var server = new Server({
dirName: __dirname,
enableSocket: true
})
.setupApp()
.setupRoutes();
return server.http;
}).listen(3000, function() {
console.log('server started on 3000 port');
});
}
接下来的几天我将做更多测试,但是,如果有人可以提出一些想法,这将对您有很大帮助。
谢谢,
您可能正在寻找socket.io-redis。 http://socket.io/blog/introducing-socket-io-1-0/ (滚动到“可扩展性”)
这是有关如何使用socket.io + express创建脚手架的简短示例:
var cluster = require('cluster');
var express = require('express')
, app = express()
, server = require('http').createServer(app);
var
io = require('socket.io').listen(server)
var redis = require('socket.io-redis');
io.adapter(redis({ host: 'localhost', port: 6379 }));
var workers = process.env.WORKERS || require('os').cpus().length;
/**
* Start cluster.
*/
if (cluster.isMaster) {
/**
* Fork process.
*/
console.log('start cluster with %s workers', workers-1);
workers--;
for (var i = 0; i < workers; ++i) {
var worker = cluster.fork();
console.log('worker %s started.', worker.process.pid);
}
/**
* Restart process.
*/
cluster.on('death', function(worker) {
console.log('worker %s died. restart...', worker.process.pid);
cluster.fork();
});
} else {
server.listen(process.env.PORT || 9010);
}
Redis具有pub / sub,并且所有socket.io节点都需要订阅redis才能从通道获取所有消息。 这样,一个进程可以将消息广播到通道(发布),而所有其他进程以最小的延迟接收消息,以将消息广播到其连接的客户端(订阅)。 您甚至可以使用基于Redis的会话来扩展它。
我认为您所指的集群模块有点误导。 据我所知,它有助于创建单个子流程,但不会“同步”多个节点之间的通道。 如果您的客户不需要与他人交流,那就很好。 如果要向所有节点上的所有已连接客户端广播消息,则需要redis模块。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.