mirror of
https://github.com/NodeBB/NodeBB.git
synced 2025-12-20 15:30:39 +01:00
Allow running as a cluster without Redis (#6233)
* [database/*] Allow databases other than Redis to provide pubsub for clustering if Redis is not present * [pubsub] Delay messages sent before the database is ready until the database is ready. * [pubsub] Restore old behavior of not using the database in non-clustered NodeBB instances. See comment: https://github.com/NodeBB/NodeBB/pull/6233#issuecomment-357814968
This commit is contained in:
committed by
Barış Soner Uşaklı
parent
c20aca8933
commit
e85aabbe74
@@ -58,6 +58,7 @@
|
|||||||
"mongodb": "2.2.33",
|
"mongodb": "2.2.33",
|
||||||
"morgan": "^1.9.0",
|
"morgan": "^1.9.0",
|
||||||
"mousetrap": "^1.6.1",
|
"mousetrap": "^1.6.1",
|
||||||
|
"mubsub": "^1.4.0",
|
||||||
"nconf": "^0.9.1",
|
"nconf": "^0.9.1",
|
||||||
"nodebb-plugin-composer-default": "6.0.8",
|
"nodebb-plugin-composer-default": "6.0.8",
|
||||||
"nodebb-plugin-dbsearch": "2.0.9",
|
"nodebb-plugin-dbsearch": "2.0.9",
|
||||||
@@ -89,6 +90,7 @@
|
|||||||
"serve-favicon": "^2.4.5",
|
"serve-favicon": "^2.4.5",
|
||||||
"sitemap": "^1.13.0",
|
"sitemap": "^1.13.0",
|
||||||
"socket.io": "2.0.4",
|
"socket.io": "2.0.4",
|
||||||
|
"socket.io-adapter-mongo": "^2.0.1",
|
||||||
"socket.io-client": "2.0.4",
|
"socket.io-client": "2.0.4",
|
||||||
"socket.io-redis": "5.2.0",
|
"socket.io-redis": "5.2.0",
|
||||||
"socketio-wildcard": "2.0.0",
|
"socketio-wildcard": "2.0.0",
|
||||||
|
|||||||
@@ -61,11 +61,7 @@ mongoModule.questions = [
|
|||||||
mongoModule.helpers = mongoModule.helpers || {};
|
mongoModule.helpers = mongoModule.helpers || {};
|
||||||
mongoModule.helpers.mongo = require('./mongo/helpers');
|
mongoModule.helpers.mongo = require('./mongo/helpers');
|
||||||
|
|
||||||
mongoModule.init = function (callback) {
|
function getConnectionString() {
|
||||||
callback = callback || function () { };
|
|
||||||
|
|
||||||
var mongoClient = require('mongodb').MongoClient;
|
|
||||||
|
|
||||||
var usernamePassword = '';
|
var usernamePassword = '';
|
||||||
if (nconf.get('mongo:username') && nconf.get('mongo:password')) {
|
if (nconf.get('mongo:username') && nconf.get('mongo:password')) {
|
||||||
usernamePassword = nconf.get('mongo:username') + ':' + encodeURIComponent(nconf.get('mongo:password')) + '@';
|
usernamePassword = nconf.get('mongo:username') + ':' + encodeURIComponent(nconf.get('mongo:password')) + '@';
|
||||||
@@ -92,7 +88,15 @@ mongoModule.init = function (callback) {
|
|||||||
servers.push(hosts[i] + ':' + ports[i]);
|
servers.push(hosts[i] + ':' + ports[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
var connString = nconf.get('mongo:uri') || 'mongodb://' + usernamePassword + servers.join() + '/' + nconf.get('mongo:database');
|
return nconf.get('mongo:uri') || 'mongodb://' + usernamePassword + servers.join() + '/' + nconf.get('mongo:database');
|
||||||
|
}
|
||||||
|
|
||||||
|
mongoModule.init = function (callback) {
|
||||||
|
callback = callback || function () { };
|
||||||
|
|
||||||
|
var mongoClient = require('mongodb').MongoClient;
|
||||||
|
|
||||||
|
var connString = getConnectionString();
|
||||||
|
|
||||||
var connOptions = {
|
var connOptions = {
|
||||||
poolSize: 10,
|
poolSize: 10,
|
||||||
@@ -118,6 +122,7 @@ mongoModule.init = function (callback) {
|
|||||||
require('./mongo/sets')(db, mongoModule);
|
require('./mongo/sets')(db, mongoModule);
|
||||||
require('./mongo/sorted')(db, mongoModule);
|
require('./mongo/sorted')(db, mongoModule);
|
||||||
require('./mongo/list')(db, mongoModule);
|
require('./mongo/list')(db, mongoModule);
|
||||||
|
require('./mongo/pubsub')(db, mongoModule);
|
||||||
callback();
|
callback();
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
@@ -262,3 +267,8 @@ mongoModule.close = function (callback) {
|
|||||||
callback = callback || function () {};
|
callback = callback || function () {};
|
||||||
db.close(callback);
|
db.close(callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
mongoModule.socketAdapter = function () {
|
||||||
|
var mongoAdapter = require('socket.io-adapter-mongo');
|
||||||
|
return mongoAdapter(getConnectionString());
|
||||||
|
};
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ module.exports = function (db, module) {
|
|||||||
|
|
||||||
var LRU = require('lru-cache');
|
var LRU = require('lru-cache');
|
||||||
var _ = require('lodash');
|
var _ = require('lodash');
|
||||||
var pubsub = require('../../pubsub');
|
|
||||||
|
|
||||||
var cache = LRU({
|
var cache = LRU({
|
||||||
max: 10000,
|
max: 10000,
|
||||||
@@ -17,24 +16,6 @@ module.exports = function (db, module) {
|
|||||||
cache.hits = 0;
|
cache.hits = 0;
|
||||||
module.objectCache = cache;
|
module.objectCache = cache;
|
||||||
|
|
||||||
pubsub.on('mongo:hash:cache:del', function (key) {
|
|
||||||
cache.del(key);
|
|
||||||
});
|
|
||||||
|
|
||||||
pubsub.on('mongo:hash:cache:reset', function () {
|
|
||||||
cache.reset();
|
|
||||||
});
|
|
||||||
|
|
||||||
module.delObjectCache = function (key) {
|
|
||||||
pubsub.publish('mongo:hash:cache:del', key);
|
|
||||||
cache.del(key);
|
|
||||||
};
|
|
||||||
|
|
||||||
module.resetObjectCache = function () {
|
|
||||||
pubsub.publish('mongo:hash:cache:reset');
|
|
||||||
cache.reset();
|
|
||||||
};
|
|
||||||
|
|
||||||
module.setObject = function (key, data, callback) {
|
module.setObject = function (key, data, callback) {
|
||||||
callback = callback || helpers.noop;
|
callback = callback || helpers.noop;
|
||||||
if (!key || !data) {
|
if (!key || !data) {
|
||||||
|
|||||||
34
src/database/mongo/pubsub.js
Normal file
34
src/database/mongo/pubsub.js
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
var nconf = require('nconf');
|
||||||
|
|
||||||
|
module.exports = function (db, mongoModule) {
|
||||||
|
var pubsub;
|
||||||
|
|
||||||
|
if (!nconf.get('redis')) {
|
||||||
|
var mubsub = require('mubsub');
|
||||||
|
var client = mubsub(db);
|
||||||
|
pubsub = client.channel('pubsub');
|
||||||
|
mongoModule.pubsub = pubsub;
|
||||||
|
} else {
|
||||||
|
pubsub = require('../../pubsub');
|
||||||
|
}
|
||||||
|
|
||||||
|
pubsub.on('mongo:hash:cache:del', function (key) {
|
||||||
|
mongoModule.objectCache.del(key);
|
||||||
|
});
|
||||||
|
|
||||||
|
pubsub.on('mongo:hash:cache:reset', function () {
|
||||||
|
mongoModule.objectCache.reset();
|
||||||
|
});
|
||||||
|
|
||||||
|
mongoModule.delObjectCache = function (key) {
|
||||||
|
pubsub.publish('mongo:hash:cache:del', key);
|
||||||
|
mongoModule.objectCache.del(key);
|
||||||
|
};
|
||||||
|
|
||||||
|
mongoModule.resetObjectCache = function () {
|
||||||
|
pubsub.publish('mongo:hash:cache:reset');
|
||||||
|
mongoModule.objectCache.reset();
|
||||||
|
};
|
||||||
|
};
|
||||||
@@ -50,6 +50,7 @@ redisModule.init = function (callback) {
|
|||||||
require('./redis/sets')(redisClient, redisModule);
|
require('./redis/sets')(redisClient, redisModule);
|
||||||
require('./redis/sorted')(redisClient, redisModule);
|
require('./redis/sorted')(redisClient, redisModule);
|
||||||
require('./redis/list')(redisClient, redisModule);
|
require('./redis/list')(redisClient, redisModule);
|
||||||
|
require('./redis/pubsub')(redisClient, redisModule);
|
||||||
|
|
||||||
callback();
|
callback();
|
||||||
});
|
});
|
||||||
@@ -169,5 +170,16 @@ redisModule.info = function (cxn, callback) {
|
|||||||
], callback);
|
], callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
redisModule.socketAdapter = function () {
|
||||||
|
var redisAdapter = require('socket.io-redis');
|
||||||
|
var pub = redisModule.connect();
|
||||||
|
var sub = redisModule.connect();
|
||||||
|
return redisAdapter({
|
||||||
|
key: 'db:' + nconf.get('redis:database') + ':adapter_key',
|
||||||
|
pubClient: pub,
|
||||||
|
subClient: sub,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
redisModule.helpers = redisModule.helpers || {};
|
redisModule.helpers = redisModule.helpers || {};
|
||||||
redisModule.helpers.redis = require('./redis/helpers');
|
redisModule.helpers.redis = require('./redis/helpers');
|
||||||
|
|||||||
40
src/database/redis/pubsub.js
Normal file
40
src/database/redis/pubsub.js
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
var nconf = require('nconf');
|
||||||
|
var util = require('util');
|
||||||
|
var winston = require('winston');
|
||||||
|
var EventEmitter = require('events').EventEmitter;
|
||||||
|
|
||||||
|
var channelName;
|
||||||
|
|
||||||
|
var PubSub = function (redisModule) {
|
||||||
|
var self = this;
|
||||||
|
var subClient = redisModule.connect();
|
||||||
|
this.pubClient = redisModule.connect();
|
||||||
|
|
||||||
|
channelName = 'db:' + nconf.get('redis:database') + 'pubsub_channel';
|
||||||
|
subClient.subscribe(channelName);
|
||||||
|
|
||||||
|
subClient.on('message', function (channel, message) {
|
||||||
|
if (channel !== channelName) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
var msg = JSON.parse(message);
|
||||||
|
self.emit(msg.event, msg.data);
|
||||||
|
} catch (err) {
|
||||||
|
winston.error(err.stack);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
util.inherits(PubSub, EventEmitter);
|
||||||
|
|
||||||
|
PubSub.prototype.publish = function (event, data) {
|
||||||
|
this.pubClient.publish(channelName, JSON.stringify({ event: event, data: data }));
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = function (redisClient, redisModule) {
|
||||||
|
redisModule.pubsub = new PubSub(redisModule);
|
||||||
|
};
|
||||||
@@ -1,48 +1,70 @@
|
|||||||
|
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
var nconf = require('nconf');
|
var nconf = require('nconf');
|
||||||
var util = require('util');
|
|
||||||
var winston = require('winston');
|
|
||||||
var EventEmitter = require('events').EventEmitter;
|
|
||||||
|
|
||||||
var channelName;
|
var real;
|
||||||
|
var fake = {
|
||||||
var PubSub = function () {
|
publishQueue: [],
|
||||||
var self = this;
|
publish: function (event, data) {
|
||||||
if (nconf.get('redis')) {
|
fake.publishQueue.push({ event: event, data: data });
|
||||||
var redis = require('./database/redis');
|
},
|
||||||
var subClient = redis.connect();
|
listenQueue: {},
|
||||||
this.pubClient = redis.connect();
|
on: function (event, callback) {
|
||||||
|
if (!Object.prototype.hasOwnProperty.call(fake.listenQueue, event)) {
|
||||||
channelName = 'db:' + nconf.get('redis:database') + 'pubsub_channel';
|
fake.listenQueue[event] = [];
|
||||||
subClient.subscribe(channelName);
|
}
|
||||||
|
fake.listenQueue[event].push(callback);
|
||||||
subClient.on('message', function (channel, message) {
|
},
|
||||||
if (channel !== channelName) {
|
removeAllListeners: function (event) {
|
||||||
return;
|
delete fake.listenQueue[event];
|
||||||
}
|
},
|
||||||
|
|
||||||
try {
|
|
||||||
var msg = JSON.parse(message);
|
|
||||||
self.emit(msg.event, msg.data);
|
|
||||||
} catch (err) {
|
|
||||||
winston.error(err.stack);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
util.inherits(PubSub, EventEmitter);
|
function get() {
|
||||||
|
if (real) {
|
||||||
|
return real;
|
||||||
|
}
|
||||||
|
|
||||||
PubSub.prototype.publish = function (event, data) {
|
var pubsub;
|
||||||
if (this.pubClient) {
|
|
||||||
this.pubClient.publish(channelName, JSON.stringify({ event: event, data: data }));
|
if (nconf.get('isCluster') === 'false') {
|
||||||
|
var EventEmitter = require('events');
|
||||||
|
pubsub = new EventEmitter();
|
||||||
|
pubsub.publish = pubsub.emit.bind(pubsub);
|
||||||
|
} else if (nconf.get('redis')) {
|
||||||
|
pubsub = require('./database/redis').pubsub;
|
||||||
} else {
|
} else {
|
||||||
this.emit(event, data);
|
pubsub = require('./database').pubsub;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!pubsub) {
|
||||||
|
return fake;
|
||||||
|
}
|
||||||
|
|
||||||
|
Object.keys(fake.listenQueue).forEach(function (event) {
|
||||||
|
fake.listenQueue[event].forEach(function (callback) {
|
||||||
|
pubsub.on(event, callback);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
fake.publishQueue.forEach(function (msg) {
|
||||||
|
pubsub.publish(msg.event, msg.data);
|
||||||
|
});
|
||||||
|
|
||||||
|
real = pubsub;
|
||||||
|
fake = null;
|
||||||
|
|
||||||
|
return pubsub;
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
publish: function (event, data) {
|
||||||
|
get().publish(event, data);
|
||||||
|
},
|
||||||
|
on: function (event, callback) {
|
||||||
|
get().on(event, callback);
|
||||||
|
},
|
||||||
|
removeAllListeners: function (event) {
|
||||||
|
get().removeAllListeners(event);
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
var pubsub = new PubSub();
|
|
||||||
|
|
||||||
module.exports = pubsub;
|
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ Sockets.init = function (server) {
|
|||||||
path: nconf.get('relative_path') + '/socket.io',
|
path: nconf.get('relative_path') + '/socket.io',
|
||||||
});
|
});
|
||||||
|
|
||||||
addRedisAdapter(io);
|
io.adapter(nconf.get('redis') ? require('../database/redis').socketAdapter() : db.socketAdapter());
|
||||||
|
|
||||||
io.use(socketioWildcard);
|
io.use(socketioWildcard);
|
||||||
io.use(authorize);
|
io.use(authorize);
|
||||||
@@ -212,22 +212,6 @@ function authorize(socket, callback) {
|
|||||||
], callback);
|
], callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
function addRedisAdapter(io) {
|
|
||||||
if (nconf.get('redis')) {
|
|
||||||
var redisAdapter = require('socket.io-redis');
|
|
||||||
var redis = require('../database/redis');
|
|
||||||
var pub = redis.connect();
|
|
||||||
var sub = redis.connect();
|
|
||||||
io.adapter(redisAdapter({
|
|
||||||
key: 'db:' + nconf.get('redis:database') + ':adapter_key',
|
|
||||||
pubClient: pub,
|
|
||||||
subClient: sub,
|
|
||||||
}));
|
|
||||||
} else if (nconf.get('isCluster') === 'true') {
|
|
||||||
winston.warn('[socket.io] Clustering detected, you are advised to configure Redis as a websocket store.');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Sockets.in = function (room) {
|
Sockets.in = function (room) {
|
||||||
return io.in(room);
|
return io.in(room);
|
||||||
};
|
};
|
||||||
|
|||||||
Reference in New Issue
Block a user