1 | ;
|
2 |
|
3 | var _toConsumableArray2 = require('babel-runtime/helpers/toConsumableArray');
|
4 |
|
5 | var _toConsumableArray3 = _interopRequireDefault(_toConsumableArray2);
|
6 |
|
7 | var _getIterator2 = require('babel-runtime/core-js/get-iterator');
|
8 |
|
9 | var _getIterator3 = _interopRequireDefault(_getIterator2);
|
10 |
|
11 | var _stringify = require('babel-runtime/core-js/json/stringify');
|
12 |
|
13 | var _stringify2 = _interopRequireDefault(_stringify);
|
14 |
|
15 | var _slicedToArray2 = require('babel-runtime/helpers/slicedToArray');
|
16 |
|
17 | var _slicedToArray3 = _interopRequireDefault(_slicedToArray2);
|
18 |
|
19 | var _getPrototypeOf = require('babel-runtime/core-js/object/get-prototype-of');
|
20 |
|
21 | var _getPrototypeOf2 = _interopRequireDefault(_getPrototypeOf);
|
22 |
|
23 | var _possibleConstructorReturn2 = require('babel-runtime/helpers/possibleConstructorReturn');
|
24 |
|
25 | var _possibleConstructorReturn3 = _interopRequireDefault(_possibleConstructorReturn2);
|
26 |
|
27 | var _inherits2 = require('babel-runtime/helpers/inherits');
|
28 |
|
29 | var _inherits3 = _interopRequireDefault(_inherits2);
|
30 |
|
31 | var _classCallCheck2 = require('babel-runtime/helpers/classCallCheck');
|
32 |
|
33 | var _classCallCheck3 = _interopRequireDefault(_classCallCheck2);
|
34 |
|
35 | var _createClass2 = require('babel-runtime/helpers/createClass');
|
36 |
|
37 | var _createClass3 = _interopRequireDefault(_createClass2);
|
38 |
|
39 | function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
40 |
|
41 | var ChatServiceError = require('./ChatServiceError');
|
42 | var Promise = require('bluebird');
|
43 | var Redis = require('ioredis');
|
44 | var Room = require('./Room');
|
45 | var User = require('./User');
|
46 | var _ = require('lodash');
|
47 | var promiseRetry = require('promise-retry');
|
48 | var uid = require('uid-safe');
|
49 |
|
50 | var _require = require('es6-mixin');
|
51 |
|
52 | var mixin = _require.mixin;
|
53 |
|
54 |
|
55 | var namespace = 'chatservice';
|
56 |
|
57 | function initSet(redis, set, values) {
|
58 | return redis.del(set).then(function () {
|
59 | if (!values) {
|
60 | return Promise.resolve();
|
61 | } else {
|
62 | return redis.sadd(set, values);
|
63 | }
|
64 | });
|
65 | }
|
66 |
|
67 | // State init/remove operations.
|
68 |
|
69 | var StateOperations = function () {
|
70 | function StateOperations(name, exitsErrorName, redis, makeKeyName, stateReset) {
|
71 | (0, _classCallCheck3.default)(this, StateOperations);
|
72 |
|
73 | this.name = name;
|
74 | this.exitsErrorName = exitsErrorName;
|
75 | this.redis = redis;
|
76 | this.makeKeyName = makeKeyName;
|
77 | this.stateReset = stateReset;
|
78 | }
|
79 |
|
80 | (0, _createClass3.default)(StateOperations, [{
|
81 | key: 'initState',
|
82 | value: function initState(state) {
|
83 | var _this = this;
|
84 |
|
85 | return this.redis.setnx(this.makeKeyName('exists'), true).then(function (isnew) {
|
86 | if (!isnew) {
|
87 | var error = new ChatServiceError(_this.exitsErrorName, _this.name);
|
88 | return Promise.reject(error);
|
89 | } else {
|
90 | return Promise.resolve();
|
91 | }
|
92 | }).then(function () {
|
93 | return _this.stateReset(state);
|
94 | }).then(function () {
|
95 | return _this.redis.setnx(_this.makeKeyName('isInit'), true);
|
96 | });
|
97 | }
|
98 | }, {
|
99 | key: 'removeState',
|
100 | value: function removeState() {
|
101 | var _this2 = this;
|
102 |
|
103 | return this.stateReset().then(function () {
|
104 | return _this2.redis.del(_this2.makeKeyName('exists'), _this2.makeKeyName('isInit'));
|
105 | });
|
106 | }
|
107 | }, {
|
108 | key: 'startRemoving',
|
109 | value: function startRemoving() {
|
110 | return this.redis.del(this.makeKeyName('isInit'));
|
111 | }
|
112 | }]);
|
113 | return StateOperations;
|
114 | }();
|
115 |
|
116 | // Redis lock operations.
|
117 |
|
118 |
|
119 | var LockOperations = function () {
|
120 | function LockOperations(redis) {
|
121 | (0, _classCallCheck3.default)(this, LockOperations);
|
122 |
|
123 | this.redis = redis;
|
124 | }
|
125 |
|
126 | (0, _createClass3.default)(LockOperations, [{
|
127 | key: 'lock',
|
128 | value: function lock(key, val, ttl) {
|
129 | var _this3 = this;
|
130 |
|
131 | return promiseRetry({ minTimeout: 100, retries: 10, factor: 1.5, randomize: true }, function (retry, n) {
|
132 | return _this3.redis.set(key, val, 'NX', 'PX', ttl).then(function (res) {
|
133 | if (!res) {
|
134 | var error = new ChatServiceError('timeout');
|
135 | return retry(error);
|
136 | } else {
|
137 | return null;
|
138 | }
|
139 | }).catch(retry);
|
140 | });
|
141 | }
|
142 | }, {
|
143 | key: 'unlock',
|
144 | value: function unlock(key, val) {
|
145 | return this.redis.unlock(key, val);
|
146 | }
|
147 | }]);
|
148 | return LockOperations;
|
149 | }();
|
150 |
|
151 | // Redis scripts.
|
152 |
|
153 |
|
154 | var luaCommands = {
|
155 | unlock: {
|
156 | numberOfKeys: 1,
|
157 | lua: '\nif redis.call("get",KEYS[1]) == ARGV[1] then\n return redis.call("del",KEYS[1])\nelse\n return 0\nend'
|
158 | },
|
159 |
|
160 | messageAdd: {
|
161 | numberOfKeys: 5,
|
162 | lua: '\nlocal msg = ARGV[1]\nlocal ts = ARGV[2]\n\nlocal lastMessageId = KEYS[1]\nlocal historyMaxSize = KEYS[2]\nlocal messagesIds = KEYS[3]\nlocal messagesTimestamps = KEYS[4]\nlocal messagesHistory = KEYS[5]\n\nlocal id = tonumber(redis.call(\'INCR\', lastMessageId))\nlocal maxsz = tonumber(redis.call(\'GET\', historyMaxSize))\n\nredis.call(\'LPUSH\', messagesIds, id)\nredis.call(\'LPUSH\', messagesTimestamps, ts)\nredis.call(\'LPUSH\', messagesHistory, msg)\n\nlocal sz = tonumber(redis.call(\'LLEN\', messagesHistory))\n\nif sz > maxsz then\n redis.call(\'RPOP\', messagesIds)\n redis.call(\'RPOP\', messagesTimestamps)\n redis.call(\'RPOP\', messagesHistory)\nend\n\nreturn {id}'
|
163 | },
|
164 |
|
165 | messagesGet: {
|
166 | numberOfKeys: 5,
|
167 | lua: '\nlocal id = ARGV[1]\nlocal maxlen = ARGV[2]\n\nlocal lastMessageId = KEYS[1]\nlocal historyMaxSize = KEYS[2]\nlocal messagesIds = KEYS[3]\nlocal messagesTimestamps = KEYS[4]\nlocal messagesHistory = KEYS[5]\n\nlocal lastid = tonumber(redis.call(\'GET\', lastMessageId))\nlocal maxsz = tonumber(redis.call(\'GET\', historyMaxSize))\nlocal id = math.min(id, lastid)\nlocal endp = lastid - id\nlocal len = math.min(maxlen, endp)\nlocal start = math.max(0, endp - len)\n\nif start >= endp then\n return {}\nend\n\nendp = endp - 1\nlocal msgs = redis.call(\'LRANGE\', messagesHistory, start, endp)\nlocal tss = redis.call(\'LRANGE\', messagesTimestamps, start, endp)\nlocal ids = redis.call(\'LRANGE\', messagesIds, start, endp)\n\nreturn {msgs, tss, ids}'
|
168 | },
|
169 |
|
170 | getSocketsToRooms: {
|
171 | numberOfKeys: 1,
|
172 | lua: '\nlocal result = {}\nlocal sockets = KEYS[1]\nlocal prefix = ARGV[1]\nlocal ids = redis.call(\'HKEYS\', sockets)\n\nif table.getn(ids) == 0 then\n local jsonResult = cjson.encode(cjson.null)\n return {jsonResult}\nend\n\nfor i, id in pairs(ids) do\n local joined = redis.call(\'SMEMBERS\', prefix .. id)\n result[id] = joined\nend\n\nlocal jsonResult = cjson.encode(result)\nreturn {jsonResult}'
|
173 | },
|
174 |
|
175 | removeAllSocketsFromRoom: {
|
176 | numberOfKeys: 1,
|
177 | lua: '\nlocal room = KEYS[1]\nlocal prefix = ARGV[1]\nlocal roomName = ARGV[2]\nlocal ids = redis.call(\'SMEMBERS\', room)\n\nif table.getn(ids) == 0 then\n local jsonResult = cjson.encode(cjson.null)\n return {jsonResult}\nend\n\nredis.call(\'DEL\', room)\n\nfor i, id in pairs(ids) do\n redis.call(\'SREM\', prefix .. id, roomName)\nend\n\nlocal jsonResult = cjson.encode(ids)\nreturn {jsonResult}'
|
178 | },
|
179 |
|
180 | removeSocket: {
|
181 | numberOfKeys: 2,
|
182 | lua: '\nlocal id = KEYS[1]\nlocal sockets = KEYS[2]\nlocal prefix = ARGV[1]\nlocal socketid = ARGV[2]\n\nlocal rooms = redis.call(\'SMEMBERS\', id)\nredis.call(\'DEL\', id)\n\nredis.call(\'HDEL\', sockets, socketid)\nlocal nconnected = redis.call(\'HLEN\', sockets)\n\nlocal removedRooms = {}\nlocal joinedSockets = {}\n\nfor i, room in pairs(rooms) do\n local ismember = redis.call(\'SISMEMBER\', prefix .. room, socketid)\n if ismember == 1 then\n redis.call(\'SREM\', prefix .. room, socketid)\n local njoined = redis.call(\'SCARD\', prefix .. room)\n table.insert(removedRooms, room)\n table.insert(joinedSockets, njoined)\n end\nend\n\nif table.getn(removedRooms) == 0 or table.getn(rooms) == 0 then\n local jsonResult = cjson.encode({cjson.null, cjson.null, nconnected})\n return {jsonResult}\nend\n\nlocal jsonResult = cjson.encode({removedRooms, joinedSockets, nconnected})\nreturn {jsonResult}'
|
183 | }
|
184 |
|
185 | };
|
186 |
|
187 | // Implements state API lists management.
|
188 |
|
189 | var ListsStateRedis = function () {
|
190 | function ListsStateRedis() {
|
191 | (0, _classCallCheck3.default)(this, ListsStateRedis);
|
192 | }
|
193 |
|
194 | (0, _createClass3.default)(ListsStateRedis, [{
|
195 | key: 'makeKeyName',
|
196 | value: function makeKeyName(keyName) {
|
197 | return namespace + ':' + this.prefix + ':{' + this.name + '}:' + keyName;
|
198 | }
|
199 | }, {
|
200 | key: 'checkList',
|
201 | value: function checkList(listName, num, limit) {
|
202 | if (!this.hasList(listName)) {
|
203 | var error = new ChatServiceError('noList', listName);
|
204 | return Promise.reject(error);
|
205 | }
|
206 | if (listName === 'userlist') {
|
207 | return Promise.resolve();
|
208 | }
|
209 | return this.redis.scard(listName).then(function (sz) {
|
210 | if (sz + num > limit) {
|
211 | var _error = new ChatServiceError('listLimitExceeded', listName);
|
212 | return Promise.reject(_error);
|
213 | } else {
|
214 | return Promise.resolve();
|
215 | }
|
216 | });
|
217 | }
|
218 | }, {
|
219 | key: 'addToList',
|
220 | value: function addToList(listName, elems, limit) {
|
221 | var _this4 = this;
|
222 |
|
223 | var num = elems.length;
|
224 | return this.checkList(listName, num, limit).then(function () {
|
225 | return _this4.redis.sadd(_this4.makeKeyName(listName), elems);
|
226 | });
|
227 | }
|
228 | }, {
|
229 | key: 'removeFromList',
|
230 | value: function removeFromList(listName, elems) {
|
231 | var _this5 = this;
|
232 |
|
233 | return this.checkList(listName).then(function () {
|
234 | return _this5.redis.srem(_this5.makeKeyName(listName), elems);
|
235 | });
|
236 | }
|
237 | }, {
|
238 | key: 'getList',
|
239 | value: function getList(listName) {
|
240 | var _this6 = this;
|
241 |
|
242 | return this.checkList(listName).then(function () {
|
243 | return _this6.redis.smembers(_this6.makeKeyName(listName));
|
244 | });
|
245 | }
|
246 | }, {
|
247 | key: 'hasInList',
|
248 | value: function hasInList(listName, elem) {
|
249 | var _this7 = this;
|
250 |
|
251 | return this.checkList(listName).then(function () {
|
252 | return _this7.redis.sismember(_this7.makeKeyName(listName), elem);
|
253 | }).then(function (data) {
|
254 | return Promise.resolve(Boolean(data));
|
255 | });
|
256 | }
|
257 | }, {
|
258 | key: 'whitelistOnlySet',
|
259 | value: function whitelistOnlySet(mode) {
|
260 | var whitelistOnly = mode ? true : '';
|
261 | return this.redis.set(this.makeKeyName('whitelistMode'), whitelistOnly);
|
262 | }
|
263 | }, {
|
264 | key: 'whitelistOnlyGet',
|
265 | value: function whitelistOnlyGet() {
|
266 | return this.redis.get(this.makeKeyName('whitelistMode')).then(function (data) {
|
267 | return Promise.resolve(Boolean(data));
|
268 | });
|
269 | }
|
270 | }]);
|
271 | return ListsStateRedis;
|
272 | }();
|
273 |
|
274 | // Implements room state API.
|
275 |
|
276 |
|
277 | var RoomStateRedis = function (_ListsStateRedis) {
|
278 | (0, _inherits3.default)(RoomStateRedis, _ListsStateRedis);
|
279 |
|
280 | function RoomStateRedis(server, roomName) {
|
281 | (0, _classCallCheck3.default)(this, RoomStateRedis);
|
282 |
|
283 | var _this8 = (0, _possibleConstructorReturn3.default)(this, (0, _getPrototypeOf2.default)(RoomStateRedis).call(this));
|
284 |
|
285 | _this8.server = server;
|
286 | _this8.roomName = roomName;
|
287 | _this8.name = _this8.roomName;
|
288 | _this8.historyMaxGetMessages = _this8.server.historyMaxGetMessages;
|
289 | _this8.redis = _this8.server.redis;
|
290 | _this8.exitsErrorName = 'roomExists';
|
291 | _this8.prefix = 'rooms';
|
292 | mixin(_this8, StateOperations, _this8.name, _this8.exitsErrorName, _this8.redis, _this8.makeKeyName.bind(_this8), _this8.stateReset.bind(_this8));
|
293 | return _this8;
|
294 | }
|
295 |
|
296 | (0, _createClass3.default)(RoomStateRedis, [{
|
297 | key: 'stateReset',
|
298 | value: function stateReset(state) {
|
299 | state = state || {};
|
300 | var _state = state;
|
301 | var whitelist = _state.whitelist;
|
302 | var blacklist = _state.blacklist;
|
303 | var adminlist = _state.adminlist;
|
304 | var whitelistOnly = _state.whitelistOnly;
|
305 | var owner = _state.owner;
|
306 | var historyMaxSize = _state.historyMaxSize;
|
307 | var _state$enableAccessLi = _state.enableAccessListsUpdates;
|
308 | var enableAccessListsUpdates = _state$enableAccessLi === undefined ? this.server.enableAccessListsUpdates : _state$enableAccessLi;
|
309 | var _state$enableUserlist = _state.enableUserlistUpdates;
|
310 | var enableUserlistUpdates = _state$enableUserlist === undefined ? this.server.enableUserlistUpdates : _state$enableUserlist;
|
311 |
|
312 | if (!owner) {
|
313 | owner = '';
|
314 | }
|
315 | return Promise.all([initSet(this.redis, this.makeKeyName('whitelist'), whitelist), initSet(this.redis, this.makeKeyName('blacklist'), blacklist), initSet(this.redis, this.makeKeyName('adminlist'), adminlist), initSet(this.redis, this.makeKeyName('userlist'), null), this.redis.del(this.makeKeyName('messagesHistory')), this.redis.del(this.makeKeyName('messagesTimestamps')), this.redis.del(this.makeKeyName('messagesIds')), this.redis.del(this.makeKeyName('usersseen')), this.redis.set(this.makeKeyName('lastMessageId'), 0), this.redis.set(this.makeKeyName('owner'), owner), this.whitelistOnlySet(whitelistOnly), this.accessListsUpdatesSet(enableAccessListsUpdates), this.userlistUpdatesSet(enableUserlistUpdates), this.historyMaxSizeSet(historyMaxSize)]).return();
|
316 | }
|
317 | }, {
|
318 | key: 'hasList',
|
319 | value: function hasList(listName) {
|
320 | return listName === 'adminlist' || listName === 'whitelist' || listName === 'blacklist' || listName === 'userlist';
|
321 | }
|
322 | }, {
|
323 | key: 'ownerGet',
|
324 | value: function ownerGet() {
|
325 | return this.redis.get(this.makeKeyName('owner'));
|
326 | }
|
327 | }, {
|
328 | key: 'ownerSet',
|
329 | value: function ownerSet(owner) {
|
330 | return this.redis.set(this.makeKeyName('owner'), owner);
|
331 | }
|
332 | }, {
|
333 | key: 'accessListsUpdatesSet',
|
334 | value: function accessListsUpdatesSet(enableAccessListsUpdates) {
|
335 | enableAccessListsUpdates = enableAccessListsUpdates ? true : '';
|
336 | return this.redis.set(this.makeKeyName('enableAccessListsUpdates'), enableAccessListsUpdates);
|
337 | }
|
338 | }, {
|
339 | key: 'accessListsUpdatesGet',
|
340 | value: function accessListsUpdatesGet() {
|
341 | return this.redis.get(this.makeKeyName('enableAccessListsUpdates')).then(function (data) {
|
342 | return Promise.resolve(Boolean(data));
|
343 | });
|
344 | }
|
345 | }, {
|
346 | key: 'userlistUpdatesSet',
|
347 | value: function userlistUpdatesSet(enableUserlistUpdates) {
|
348 | enableUserlistUpdates = enableUserlistUpdates ? true : '';
|
349 | return this.redis.set(this.makeKeyName('enableUserlistUpdates'), enableUserlistUpdates);
|
350 | }
|
351 | }, {
|
352 | key: 'userlistUpdatesGet',
|
353 | value: function userlistUpdatesGet() {
|
354 | return this.redis.get(this.makeKeyName('enableUserlistUpdates')).then(function (data) {
|
355 | return Promise.resolve(Boolean(data));
|
356 | });
|
357 | }
|
358 | }, {
|
359 | key: 'historyMaxSizeSet',
|
360 | value: function historyMaxSizeSet(historyMaxSize) {
|
361 | var limit = historyMaxSize;
|
362 | if (!(_.isNumber(historyMaxSize) && historyMaxSize >= 0)) {
|
363 | limit = this.server.historyMaxSize;
|
364 | }
|
365 | if (limit === 0) {
|
366 | return this.redis.multi().set(this.makeKeyName('historyMaxSize'), limit).del(this.makeKeyName('messagesHistory')).del(this.makeKeyName('messagesTimestamps')).del(this.makeKeyName('messagesIds')).exec();
|
367 | } else {
|
368 | var last = limit - 1;
|
369 | return this.redis.multi().set(this.makeKeyName('historyMaxSize'), limit).ltrim(this.makeKeyName('messagesHistory'), 0, last).ltrim(this.makeKeyName('messagesTimestamps'), 0, last).ltrim(this.makeKeyName('messagesIds'), 0, last).exec();
|
370 | }
|
371 | }
|
372 | }, {
|
373 | key: 'historyInfo',
|
374 | value: function historyInfo() {
|
375 | var _this9 = this;
|
376 |
|
377 | return this.redis.multi().get(this.makeKeyName('historyMaxSize')).llen(this.makeKeyName('messagesHistory')).get(this.makeKeyName('lastMessageId')).exec().spread(function (_ref, _ref2, _ref3) {
|
378 | var _ref6 = (0, _slicedToArray3.default)(_ref, 2);
|
379 |
|
380 | var historyMaxSize = _ref6[1];
|
381 |
|
382 | var _ref5 = (0, _slicedToArray3.default)(_ref2, 2);
|
383 |
|
384 | var historySize = _ref5[1];
|
385 |
|
386 | var _ref4 = (0, _slicedToArray3.default)(_ref3, 2);
|
387 |
|
388 | var lastMessageId = _ref4[1];
|
389 |
|
390 | historySize = parseInt(historySize);
|
391 | historyMaxSize = parseFloat(historyMaxSize);
|
392 | lastMessageId = parseInt(lastMessageId);
|
393 | var info = { historySize: historySize,
|
394 | historyMaxSize: historyMaxSize,
|
395 | historyMaxGetMessages: _this9.historyMaxGetMessages,
|
396 | lastMessageId: lastMessageId };
|
397 | return Promise.resolve(info);
|
398 | });
|
399 | }
|
400 | }, {
|
401 | key: 'getCommonUsers',
|
402 | value: function getCommonUsers() {
|
403 | return this.redis.sdiff(this.makeKeyName('userlist'), this.makeKeyName('whitelist'), this.makeKeyName('adminlist'));
|
404 | }
|
405 | }, {
|
406 | key: 'messageAdd',
|
407 | value: function messageAdd(msg) {
|
408 | var timestamp = _.now();
|
409 | var smsg = (0, _stringify2.default)(msg);
|
410 | return this.redis.messageAdd(this.makeKeyName('lastMessageId'), this.makeKeyName('historyMaxSize'), this.makeKeyName('messagesIds'), this.makeKeyName('messagesTimestamps'), this.makeKeyName('messagesHistory'), smsg, timestamp).spread(function (id) {
|
411 | msg.id = id;
|
412 | msg.timestamp = timestamp;
|
413 | return Promise.resolve(msg);
|
414 | });
|
415 | }
|
416 | }, {
|
417 | key: 'convertMessages',
|
418 | value: function convertMessages(msgs, tss, ids) {
|
419 | var data = [];
|
420 | if (!msgs) {
|
421 | return Promise.resolve(data);
|
422 | }
|
423 | for (var idx = 0; idx < msgs.length; idx++) {
|
424 | var msg = msgs[idx];
|
425 | var obj = JSON.parse(msg, function (key, val) {
|
426 | if (val && val.type === 'Buffer') {
|
427 | return new Buffer(val.data);
|
428 | } else {
|
429 | return val;
|
430 | }
|
431 | });
|
432 | obj.timestamp = parseInt(tss[idx]);
|
433 | obj.id = parseInt(ids[idx]);
|
434 | data[idx] = obj;
|
435 | }
|
436 | return Promise.resolve(data);
|
437 | }
|
438 | }, {
|
439 | key: 'messagesGetRecent',
|
440 | value: function messagesGetRecent() {
|
441 | var _this10 = this;
|
442 |
|
443 | if (this.historyMaxGetMessages <= 0) {
|
444 | return Promise.resolve([]);
|
445 | }
|
446 | var limit = this.historyMaxGetMessages - 1;
|
447 | return this.redis.multi().lrange(this.makeKeyName('messagesHistory'), 0, limit).lrange(this.makeKeyName('messagesTimestamps'), 0, limit).lrange(this.makeKeyName('messagesIds'), 0, limit).exec().spread(function (_ref7, _ref8, _ref9) {
|
448 | var _ref12 = (0, _slicedToArray3.default)(_ref7, 2);
|
449 |
|
450 | var msgs = _ref12[1];
|
451 |
|
452 | var _ref11 = (0, _slicedToArray3.default)(_ref8, 2);
|
453 |
|
454 | var tss = _ref11[1];
|
455 |
|
456 | var _ref10 = (0, _slicedToArray3.default)(_ref9, 2);
|
457 |
|
458 | var ids = _ref10[1];
|
459 |
|
460 | return _this10.convertMessages(msgs, tss, ids);
|
461 | });
|
462 | }
|
463 | }, {
|
464 | key: 'messagesGet',
|
465 | value: function messagesGet(id) {
|
466 | var _this11 = this;
|
467 |
|
468 | var maxMessages = arguments.length <= 1 || arguments[1] === undefined ? this.historyMaxGetMessages : arguments[1];
|
469 |
|
470 | if (maxMessages <= 0) {
|
471 | return Promise.resolve([]);
|
472 | }
|
473 | id = _.max([0, id]);
|
474 | return this.redis.messagesGet(this.makeKeyName('lastMessageId'), this.makeKeyName('historyMaxSize'), this.makeKeyName('messagesIds'), this.makeKeyName('messagesTimestamps'), this.makeKeyName('messagesHistory'), id, maxMessages).spread(function (msgs, tss, ids) {
|
475 | return _this11.convertMessages(msgs, tss, ids);
|
476 | });
|
477 | }
|
478 | }, {
|
479 | key: 'userSeenGet',
|
480 | value: function userSeenGet(userName) {
|
481 | return this.redis.multi().hget(this.makeKeyName('usersseen'), userName).sismember(this.makeKeyName('userlist'), userName).exec().spread(function (_ref13, _ref14) {
|
482 | var _ref16 = (0, _slicedToArray3.default)(_ref13, 2);
|
483 |
|
484 | var ts = _ref16[1];
|
485 |
|
486 | var _ref15 = (0, _slicedToArray3.default)(_ref14, 2);
|
487 |
|
488 | var isjoined = _ref15[1];
|
489 |
|
490 | var joined = Boolean(isjoined);
|
491 | var timestamp = ts ? parseInt(ts) : null;
|
492 | return { joined: joined, timestamp: timestamp };
|
493 | });
|
494 | }
|
495 | }, {
|
496 | key: 'userSeenUpdate',
|
497 | value: function userSeenUpdate(userName) {
|
498 | var timestamp = _.now();
|
499 | return this.redis.hset(this.makeKeyName('usersseen'), userName, timestamp);
|
500 | }
|
501 | }]);
|
502 | return RoomStateRedis;
|
503 | }(ListsStateRedis);
|
504 |
|
505 | // Implements direct messaging state API.
|
506 |
|
507 |
|
508 | var DirectMessagingStateRedis = function (_ListsStateRedis2) {
|
509 | (0, _inherits3.default)(DirectMessagingStateRedis, _ListsStateRedis2);
|
510 |
|
511 | function DirectMessagingStateRedis(server, userName) {
|
512 | (0, _classCallCheck3.default)(this, DirectMessagingStateRedis);
|
513 |
|
514 | var _this12 = (0, _possibleConstructorReturn3.default)(this, (0, _getPrototypeOf2.default)(DirectMessagingStateRedis).call(this));
|
515 |
|
516 | _this12.server = server;
|
517 | _this12.userName = userName;
|
518 | _this12.name = _this12.userName;
|
519 | _this12.prefix = 'users';
|
520 | _this12.exitsErrorName = 'userExists';
|
521 | _this12.redis = _this12.server.redis;
|
522 | mixin(_this12, StateOperations, _this12.name, _this12.exitsErrorName, _this12.redis, _this12.makeKeyName.bind(_this12), _this12.stateReset.bind(_this12));
|
523 | return _this12;
|
524 | }
|
525 |
|
526 | (0, _createClass3.default)(DirectMessagingStateRedis, [{
|
527 | key: 'hasList',
|
528 | value: function hasList(listName) {
|
529 | return listName === 'whitelist' || listName === 'blacklist';
|
530 | }
|
531 | }, {
|
532 | key: 'stateReset',
|
533 | value: function stateReset(state) {
|
534 | state = state || {};
|
535 | var _state2 = state;
|
536 | var whitelist = _state2.whitelist;
|
537 | var blacklist = _state2.blacklist;
|
538 | var whitelistOnly = _state2.whitelistOnly;
|
539 |
|
540 | whitelistOnly = whitelistOnly ? true : '';
|
541 | return Promise.all([initSet(this.redis, this.makeKeyName('whitelist'), whitelist), initSet(this.redis, this.makeKeyName('blacklist'), blacklist), this.redis.set(this.makeKeyName('whitelistMode'), whitelistOnly)]).return();
|
542 | }
|
543 | }]);
|
544 | return DirectMessagingStateRedis;
|
545 | }(ListsStateRedis);
|
546 |
|
547 | // Implements user state API.
|
548 |
|
549 |
|
550 | var UserStateRedis = function () {
|
551 | function UserStateRedis(server, userName) {
|
552 | (0, _classCallCheck3.default)(this, UserStateRedis);
|
553 |
|
554 | this.server = server;
|
555 | this.userName = userName;
|
556 | this.name = this.userName;
|
557 | this.prefix = 'users';
|
558 | this.redis = this.server.redis;
|
559 | mixin(this, LockOperations, this.redis);
|
560 | }
|
561 |
|
562 | (0, _createClass3.default)(UserStateRedis, [{
|
563 | key: 'makeKeyName',
|
564 | value: function makeKeyName(keyName) {
|
565 | return namespace + ':' + this.prefix + ':{' + this.name + '}:' + keyName;
|
566 | }
|
567 | }, {
|
568 | key: 'makeSocketToRooms',
|
569 | value: function makeSocketToRooms() {
|
570 | var id = arguments.length <= 0 || arguments[0] === undefined ? '' : arguments[0];
|
571 |
|
572 | return this.makeKeyName('socketsToRooms:' + id);
|
573 | }
|
574 | }, {
|
575 | key: 'makeRoomToSockets',
|
576 | value: function makeRoomToSockets() {
|
577 | var room = arguments.length <= 0 || arguments[0] === undefined ? '' : arguments[0];
|
578 |
|
579 | return this.makeKeyName('roomsToSockets:' + room);
|
580 | }
|
581 | }, {
|
582 | key: 'makeRoomLock',
|
583 | value: function makeRoomLock(room) {
|
584 | return this.makeKeyName('roomLock:' + room);
|
585 | }
|
586 | }, {
|
587 | key: 'addSocket',
|
588 | value: function addSocket(id, uid) {
|
589 | return this.redis.multi().hset(this.makeKeyName('sockets'), id, uid).hlen(this.makeKeyName('sockets')).exec().spread(function (_, _ref17) {
|
590 | var _ref18 = (0, _slicedToArray3.default)(_ref17, 2);
|
591 |
|
592 | var nconnected = _ref18[1];
|
593 | return Promise.resolve(nconnected);
|
594 | });
|
595 | }
|
596 | }, {
|
597 | key: 'getAllSockets',
|
598 | value: function getAllSockets() {
|
599 | return this.redis.hkeys(this.makeKeyName('sockets'));
|
600 | }
|
601 | }, {
|
602 | key: 'getSocketsToInstance',
|
603 | value: function getSocketsToInstance() {
|
604 | return this.redis.hgetall(this.makeKeyName('sockets'));
|
605 | }
|
606 | }, {
|
607 | key: 'getRoomToSockets',
|
608 | value: function getRoomToSockets(roomName) {
|
609 | return this.redis.smembers(this.makeRoomToSockets(roomName));
|
610 | }
|
611 | }, {
|
612 | key: 'getSocketsToRooms',
|
613 | value: function getSocketsToRooms() {
|
614 | return this.redis.getSocketsToRooms(this.makeKeyName('sockets'), this.makeSocketToRooms()).spread(function (result) {
|
615 | var data = JSON.parse(result) || {};
|
616 | var _iteratorNormalCompletion = true;
|
617 | var _didIteratorError = false;
|
618 | var _iteratorError = undefined;
|
619 |
|
620 | try {
|
621 | for (var _iterator = (0, _getIterator3.default)(_.toPairs(data)), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {
|
622 | var _step$value = (0, _slicedToArray3.default)(_step.value, 2);
|
623 |
|
624 | var k = _step$value[0];
|
625 | var v = _step$value[1];
|
626 |
|
627 | if (_.isEmpty(v)) {
|
628 | data[k] = [];
|
629 | }
|
630 | }
|
631 | } catch (err) {
|
632 | _didIteratorError = true;
|
633 | _iteratorError = err;
|
634 | } finally {
|
635 | try {
|
636 | if (!_iteratorNormalCompletion && _iterator.return) {
|
637 | _iterator.return();
|
638 | }
|
639 | } finally {
|
640 | if (_didIteratorError) {
|
641 | throw _iteratorError;
|
642 | }
|
643 | }
|
644 | }
|
645 |
|
646 | return Promise.resolve(data);
|
647 | });
|
648 | }
|
649 | }, {
|
650 | key: 'addSocketToRoom',
|
651 | value: function addSocketToRoom(id, roomName) {
|
652 | return this.redis.multi().sadd(this.makeSocketToRooms(id), roomName).sadd(this.makeRoomToSockets(roomName), id).scard(this.makeRoomToSockets(roomName)).exec().then(function (_ref19) {
|
653 | var _ref20 = (0, _slicedToArray3.default)(_ref19, 3);
|
654 |
|
655 | var _ref20$ = (0, _slicedToArray3.default)(_ref20[2], 2);
|
656 |
|
657 | var njoined = _ref20$[1];
|
658 | return Promise.resolve(njoined);
|
659 | });
|
660 | }
|
661 | }, {
|
662 | key: 'removeSocketFromRoom',
|
663 | value: function removeSocketFromRoom(id, roomName) {
|
664 | return this.redis.multi().scard(this.makeRoomToSockets(roomName)).srem(this.makeSocketToRooms(id), roomName).srem(this.makeRoomToSockets(roomName), id).scard(this.makeRoomToSockets(roomName)).exec().then(function (_ref21) {
|
665 | var _ref22 = (0, _slicedToArray3.default)(_ref21, 4);
|
666 |
|
667 | var _ref22$ = (0, _slicedToArray3.default)(_ref22[0], 2);
|
668 |
|
669 | var wasjoined = _ref22$[1];
|
670 |
|
671 | var _ref22$2 = (0, _slicedToArray3.default)(_ref22[3], 2);
|
672 |
|
673 | var njoined = _ref22$2[1];
|
674 |
|
675 | var hasChanged = njoined !== wasjoined;
|
676 | return Promise.resolve([njoined, hasChanged]);
|
677 | });
|
678 | }
|
679 | }, {
|
680 | key: 'removeAllSocketsFromRoom',
|
681 | value: function removeAllSocketsFromRoom(roomName) {
|
682 | return this.redis.removeAllSocketsFromRoom(this.makeRoomToSockets(roomName), this.makeSocketToRooms(), roomName).spread(function (result) {
|
683 | return Promise.resolve(JSON.parse(result));
|
684 | });
|
685 | }
|
686 | }, {
|
687 | key: 'removeSocket',
|
688 | value: function removeSocket(id) {
|
689 | return this.redis.removeSocket(this.makeSocketToRooms(id), this.makeKeyName('sockets'), this.makeRoomToSockets(), id).spread(function (result) {
|
690 | return Promise.resolve(JSON.parse(result));
|
691 | });
|
692 | }
|
693 | }, {
|
694 | key: 'lockToRoom',
|
695 | value: function lockToRoom(roomName, ttl) {
|
696 | var _this13 = this;
|
697 |
|
698 | return uid(18).then(function (val) {
|
699 | var start = _.now();
|
700 | return _this13.lock(_this13.makeRoomLock(roomName), val, ttl).then(function () {
|
701 | return Promise.resolve().disposer(function () {
|
702 | if (start + ttl < _.now()) {
|
703 | _this13.server.emit('lockTimeExceeded', val, { userName: _this13.userName, roomName: roomName });
|
704 | }
|
705 | return _this13.unlock(_this13.makeRoomLock(roomName), val);
|
706 | });
|
707 | });
|
708 | });
|
709 | }
|
710 | }]);
|
711 | return UserStateRedis;
|
712 | }();
|
713 |
|
714 | // Implements global state API.
|
715 |
|
716 |
|
717 | var RedisState = function () {
|
718 | function RedisState(server, options) {
|
719 | (0, _classCallCheck3.default)(this, RedisState);
|
720 |
|
721 | this.server = server;
|
722 | this.options = options;
|
723 | this.closed = false;
|
724 | if (this.options.useCluster) {
|
725 | this.redis = new (Function.prototype.bind.apply(Redis.Cluster, [null].concat((0, _toConsumableArray3.default)(this.options.redisOptions))))();
|
726 | } else {
|
727 | var redisOptions = _.castArray(this.options.redisOptions);
|
728 | this.redis = new (Function.prototype.bind.apply(Redis, [null].concat((0, _toConsumableArray3.default)(redisOptions))))();
|
729 | }
|
730 | this.RoomState = RoomStateRedis;
|
731 | this.UserState = UserStateRedis;
|
732 | this.DirectMessagingState = DirectMessagingStateRedis;
|
733 | this.lockTTL = this.options.lockTTL || 10000;
|
734 | this.instanceUID = this.server.instanceUID;
|
735 | this.server.redis = this.redis;
|
736 | var _iteratorNormalCompletion2 = true;
|
737 | var _didIteratorError2 = false;
|
738 | var _iteratorError2 = undefined;
|
739 |
|
740 | try {
|
741 | for (var _iterator2 = (0, _getIterator3.default)(_.toPairs(luaCommands)), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {
|
742 | var _step2$value = (0, _slicedToArray3.default)(_step2.value, 2);
|
743 |
|
744 | var cmd = _step2$value[0];
|
745 | var def = _step2$value[1];
|
746 |
|
747 | this.redis.defineCommand(cmd, {
|
748 | numberOfKeys: def.numberOfKeys,
|
749 | lua: def.lua
|
750 | });
|
751 | }
|
752 | } catch (err) {
|
753 | _didIteratorError2 = true;
|
754 | _iteratorError2 = err;
|
755 | } finally {
|
756 | try {
|
757 | if (!_iteratorNormalCompletion2 && _iterator2.return) {
|
758 | _iterator2.return();
|
759 | }
|
760 | } finally {
|
761 | if (_didIteratorError2) {
|
762 | throw _iteratorError2;
|
763 | }
|
764 | }
|
765 | }
|
766 | }
|
767 |
|
768 | (0, _createClass3.default)(RedisState, [{
|
769 | key: 'makeKeyName',
|
770 | value: function makeKeyName(prefix, name, keyName) {
|
771 | return namespace + ':' + prefix + ':{' + name + '}:' + keyName;
|
772 | }
|
773 | }, {
|
774 | key: 'hasRoom',
|
775 | value: function hasRoom(name) {
|
776 | return this.redis.get(this.makeKeyName('rooms', name, 'isInit'));
|
777 | }
|
778 | }, {
|
779 | key: 'hasUser',
|
780 | value: function hasUser(name) {
|
781 | return this.redis.get(this.makeKeyName('users', name, 'isInit'));
|
782 | }
|
783 | }, {
|
784 | key: 'close',
|
785 | value: function close() {
|
786 | this.closed = true;
|
787 | return this.redis.quit().return();
|
788 | }
|
789 | }, {
|
790 | key: 'getRoom',
|
791 | value: function getRoom(name) {
|
792 | var isPredicate = arguments.length <= 1 || arguments[1] === undefined ? false : arguments[1];
|
793 |
|
794 | var room = new Room(this.server, name);
|
795 | return this.hasRoom(name).then(function (exists) {
|
796 | if (!exists) {
|
797 | if (isPredicate) {
|
798 | return Promise.resolve(null);
|
799 | } else {
|
800 | var error = new ChatServiceError('noRoom', name);
|
801 | return Promise.reject(error);
|
802 | }
|
803 | }
|
804 | return Promise.resolve(room);
|
805 | });
|
806 | }
|
807 | }, {
|
808 | key: 'addRoom',
|
809 | value: function addRoom(name, state) {
|
810 | var room = new Room(this.server, name);
|
811 | return room.initState(state).return(room);
|
812 | }
|
813 | }, {
|
814 | key: 'removeRoom',
|
815 | value: function removeRoom(name) {
|
816 | return Promise.resolve();
|
817 | }
|
818 | }, {
|
819 | key: 'addSocket',
|
820 | value: function addSocket(id, userName) {
|
821 | return this.redis.hset(this.makeKeyName('instances', this.instanceUID, 'sockets'), id, userName);
|
822 | }
|
823 | }, {
|
824 | key: 'removeSocket',
|
825 | value: function removeSocket(id) {
|
826 | return this.redis.hdel(this.makeKeyName('instances', this.instanceUID, 'sockets'), id);
|
827 | }
|
828 | }, {
|
829 | key: 'getInstanceSockets',
|
830 | value: function getInstanceSockets() {
|
831 | var uid = arguments.length <= 0 || arguments[0] === undefined ? this.instanceUID : arguments[0];
|
832 |
|
833 | return this.redis.hgetall(this.makeKeyName('instances', uid, 'sockets'));
|
834 | }
|
835 | }, {
|
836 | key: 'updateHeartbeat',
|
837 | value: function updateHeartbeat() {
|
838 | return this.redis.set(this.makeKeyName('instances', this.instanceUID, 'heartbeat'), _.now()).catchReturn();
|
839 | }
|
840 | }, {
|
841 | key: 'getInstanceHeartbeat',
|
842 | value: function getInstanceHeartbeat() {
|
843 | var uid = arguments.length <= 0 || arguments[0] === undefined ? this.instanceUID : arguments[0];
|
844 |
|
845 | return this.redis.get(this.makeKeyName('instances', uid, 'heartbeat')).then(function (ts) {
|
846 | return ts ? parseInt(ts) : null;
|
847 | });
|
848 | }
|
849 | }, {
|
850 | key: 'getOrAddUser',
|
851 | value: function getOrAddUser(name, state) {
|
852 | var user = new User(this.server, name);
|
853 | return this.hasUser(name).then(function (exists) {
|
854 | if (!exists) {
|
855 | return user.initState(state);
|
856 | } else {
|
857 | return Promise.resolve();
|
858 | }
|
859 | }).catch(ChatServiceError, function (e) {
|
860 | return user;
|
861 | }).return(user);
|
862 | }
|
863 | }, {
|
864 | key: 'getUser',
|
865 | value: function getUser(name) {
|
866 | var isPredicate = arguments.length <= 1 || arguments[1] === undefined ? false : arguments[1];
|
867 |
|
868 | var user = new User(this.server, name);
|
869 | return this.hasUser(name).then(function (exists) {
|
870 | if (!exists) {
|
871 | if (isPredicate) {
|
872 | return Promise.resolve(null);
|
873 | } else {
|
874 | var error = new ChatServiceError('noUser', name);
|
875 | return Promise.reject(error);
|
876 | }
|
877 | }
|
878 | return Promise.resolve(user);
|
879 | });
|
880 | }
|
881 | }, {
|
882 | key: 'addUser',
|
883 | value: function addUser(name, state) {
|
884 | var user = new User(this.server, name);
|
885 | return user.initState(state).return(user);
|
886 | }
|
887 | }, {
|
888 | key: 'removeUser',
|
889 | value: function removeUser(name) {
|
890 | return Promise.resolve();
|
891 | }
|
892 | }]);
|
893 | return RedisState;
|
894 | }();
|
895 |
|
896 | module.exports = RedisState;
|
897 | //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/RedisState.js"],"names":[],"mappings":"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","file":"RedisState.js","sourcesContent":["'use strict'\n\nconst ChatServiceError = require('./ChatServiceError')\nconst Promise = require('bluebird')\nconst Redis = require('ioredis')\nconst Room = require('./Room')\nconst User = require('./User')\nconst _ = require('lodash')\nconst promiseRetry = require('promise-retry')\nconst uid = require('uid-safe')\nconst { mixin } = require('es6-mixin')\n\nlet namespace = 'chatservice'\n\nfunction initSet (redis, set, values) {\n  return redis.del(set).then(() => {\n    if (!values) {\n      return Promise.resolve()\n    } else {\n      return redis.sadd(set, values)\n    }\n  })\n}\n\n\n// State init/remove operations.\nclass StateOperations {\n\n  constructor (name, exitsErrorName, redis, makeKeyName, stateReset) {\n    this.name = name\n    this.exitsErrorName = exitsErrorName\n    this.redis = redis\n    this.makeKeyName = makeKeyName\n    this.stateReset = stateReset\n  }\n\n  initState (state) {\n    return this.redis.setnx(this.makeKeyName('exists'), true).then(isnew => {\n      if (!isnew) {\n        let error = new ChatServiceError(this.exitsErrorName, this.name)\n        return Promise.reject(error)\n      } else {\n        return Promise.resolve()\n      }\n    }).then(() => this.stateReset(state))\n      .then(() => this.redis.setnx(this.makeKeyName('isInit'), true))\n  }\n\n  removeState () {\n    return this.stateReset().then(() => {\n      return this.redis.del(\n        this.makeKeyName('exists'), this.makeKeyName('isInit'))\n    })\n  }\n\n  startRemoving () {\n    return this.redis.del(this.makeKeyName('isInit'))\n  }\n\n}\n\n// Redis lock operations.\nclass LockOperations {\n\n  constructor (redis) {\n    this.redis = redis\n  }\n\n  lock (key, val, ttl) {\n    return promiseRetry(\n      {minTimeout: 100, retries: 10, factor: 1.5, randomize: true},\n      (retry, n) => {\n        return this.redis.set(key, val, 'NX', 'PX', ttl).then(res => {\n          if (!res) {\n            let error = new ChatServiceError('timeout')\n            return retry(error)\n          } else {\n            return null\n          }\n        }).catch(retry)\n      })\n  }\n\n  unlock (key, val) {\n    return this.redis.unlock(key, val)\n  }\n\n}\n\n// Redis scripts.\nlet luaCommands = {\n  unlock: {\n    numberOfKeys: 1,\n    lua: `\nif redis.call(\"get\",KEYS[1]) == ARGV[1] then\n  return redis.call(\"del\",KEYS[1])\nelse\n  return 0\nend`\n  },\n\n  messageAdd: {\n    numberOfKeys: 5,\n    lua: `\nlocal msg = ARGV[1]\nlocal ts = ARGV[2]\n\nlocal lastMessageId = KEYS[1]\nlocal historyMaxSize = KEYS[2]\nlocal messagesIds = KEYS[3]\nlocal messagesTimestamps = KEYS[4]\nlocal messagesHistory = KEYS[5]\n\nlocal id = tonumber(redis.call('INCR', lastMessageId))\nlocal maxsz = tonumber(redis.call('GET', historyMaxSize))\n\nredis.call('LPUSH', messagesIds, id)\nredis.call('LPUSH', messagesTimestamps, ts)\nredis.call('LPUSH', messagesHistory, msg)\n\nlocal sz = tonumber(redis.call('LLEN', messagesHistory))\n\nif sz > maxsz then\n  redis.call('RPOP', messagesIds)\n  redis.call('RPOP', messagesTimestamps)\n  redis.call('RPOP', messagesHistory)\nend\n\nreturn {id}`\n  },\n\n  messagesGet: {\n    numberOfKeys: 5,\n    lua: `\nlocal id = ARGV[1]\nlocal maxlen = ARGV[2]\n\nlocal lastMessageId = KEYS[1]\nlocal historyMaxSize = KEYS[2]\nlocal messagesIds = KEYS[3]\nlocal messagesTimestamps = KEYS[4]\nlocal messagesHistory = KEYS[5]\n\nlocal lastid = tonumber(redis.call('GET', lastMessageId))\nlocal maxsz = tonumber(redis.call('GET', historyMaxSize))\nlocal id = math.min(id, lastid)\nlocal endp = lastid - id\nlocal len = math.min(maxlen, endp)\nlocal start = math.max(0, endp - len)\n\nif start >= endp then\n  return {}\nend\n\nendp = endp - 1\nlocal msgs = redis.call('LRANGE', messagesHistory, start, endp)\nlocal tss = redis.call('LRANGE', messagesTimestamps, start, endp)\nlocal ids = redis.call('LRANGE', messagesIds, start, endp)\n\nreturn {msgs, tss, ids}`\n  },\n\n  getSocketsToRooms: {\n    numberOfKeys: 1,\n    lua: `\nlocal result = {}\nlocal sockets = KEYS[1]\nlocal prefix = ARGV[1]\nlocal ids = redis.call('HKEYS', sockets)\n\nif table.getn(ids) == 0 then\n  local jsonResult = cjson.encode(cjson.null)\n  return {jsonResult}\nend\n\nfor i, id in pairs(ids) do\n  local joined = redis.call('SMEMBERS', prefix .. id)\n  result[id] = joined\nend\n\nlocal jsonResult = cjson.encode(result)\nreturn {jsonResult}`\n  },\n\n  removeAllSocketsFromRoom: {\n    numberOfKeys: 1,\n    lua: `\nlocal room = KEYS[1]\nlocal prefix = ARGV[1]\nlocal roomName = ARGV[2]\nlocal ids = redis.call('SMEMBERS', room)\n\nif table.getn(ids) == 0 then\n  local jsonResult = cjson.encode(cjson.null)\n  return {jsonResult}\nend\n\nredis.call('DEL', room)\n\nfor i, id in pairs(ids) do\n  redis.call('SREM', prefix .. id, roomName)\nend\n\nlocal jsonResult = cjson.encode(ids)\nreturn {jsonResult}`\n  },\n\n  removeSocket: {\n    numberOfKeys: 2,\n    lua: `\nlocal id = KEYS[1]\nlocal sockets = KEYS[2]\nlocal prefix = ARGV[1]\nlocal socketid = ARGV[2]\n\nlocal rooms = redis.call('SMEMBERS', id)\nredis.call('DEL', id)\n\nredis.call('HDEL', sockets, socketid)\nlocal nconnected = redis.call('HLEN', sockets)\n\nlocal removedRooms = {}\nlocal joinedSockets = {}\n\nfor i, room in pairs(rooms) do\n  local ismember = redis.call('SISMEMBER', prefix .. room, socketid)\n  if ismember == 1 then\n    redis.call('SREM', prefix .. room, socketid)\n    local njoined = redis.call('SCARD', prefix .. room)\n    table.insert(removedRooms, room)\n    table.insert(joinedSockets, njoined)\n  end\nend\n\nif table.getn(removedRooms) == 0 or table.getn(rooms) == 0 then\n  local jsonResult = cjson.encode({cjson.null, cjson.null, nconnected})\n  return {jsonResult}\nend\n\nlocal jsonResult = cjson.encode({removedRooms, joinedSockets, nconnected})\nreturn {jsonResult}`\n  }\n\n}\n\n// Implements state API lists management.\nclass ListsStateRedis {\n\n  makeKeyName (keyName) {\n    return `${namespace}:${this.prefix}:{${this.name}}:${keyName}`\n  }\n\n  checkList (listName, num, limit) {\n    if (!this.hasList(listName)) {\n      let error = new ChatServiceError('noList', listName)\n      return Promise.reject(error)\n    }\n    if (listName === 'userlist') {\n      return Promise.resolve()\n    }\n    return this.redis.scard(listName).then(sz => {\n      if (sz + num > limit) {\n        let error = new ChatServiceError('listLimitExceeded', listName)\n        return Promise.reject(error)\n      } else {\n        return Promise.resolve()\n      }\n    })\n  }\n\n  addToList (listName, elems, limit) {\n    let num = elems.length\n    return this.checkList(listName, num, limit)\n      .then(() => this.redis.sadd(this.makeKeyName(listName), elems))\n  }\n\n  removeFromList (listName, elems) {\n    return this.checkList(listName)\n      .then(() => this.redis.srem(this.makeKeyName(listName), elems))\n  }\n\n  getList (listName) {\n    return this.checkList(listName)\n      .then(() => this.redis.smembers(this.makeKeyName(listName)))\n  }\n\n  hasInList (listName, elem) {\n    return this.checkList(listName)\n      .then(() => this.redis.sismember(this.makeKeyName(listName), elem))\n      .then(data => Promise.resolve(Boolean(data)))\n  }\n\n  whitelistOnlySet (mode) {\n    let whitelistOnly = mode ? true : ''\n    return this.redis.set(this.makeKeyName('whitelistMode'), whitelistOnly)\n  }\n\n  whitelistOnlyGet () {\n    return this.redis.get(this.makeKeyName('whitelistMode'))\n      .then(data => Promise.resolve(Boolean(data)))\n  }\n\n}\n\n// Implements room state API.\nclass RoomStateRedis extends ListsStateRedis {\n\n  constructor (server, roomName) {\n    super()\n    this.server = server\n    this.roomName = roomName\n    this.name = this.roomName\n    this.historyMaxGetMessages = this.server.historyMaxGetMessages\n    this.redis = this.server.redis\n    this.exitsErrorName = 'roomExists'\n    this.prefix = 'rooms'\n    mixin(this, StateOperations, this.name, this.exitsErrorName, this.redis,\n          this.makeKeyName.bind(this), this.stateReset.bind(this))\n  }\n\n  stateReset (state) {\n    state = state || {}\n    let { whitelist, blacklist, adminlist,\n          whitelistOnly, owner, historyMaxSize,\n          enableAccessListsUpdates = this.server.enableAccessListsUpdates,\n          enableUserlistUpdates = this.server.enableUserlistUpdates\n        } = state\n    if (!owner) { owner = '' }\n    return Promise.all([\n      initSet(this.redis, this.makeKeyName('whitelist'), whitelist),\n      initSet(this.redis, this.makeKeyName('blacklist'), blacklist),\n      initSet(this.redis, this.makeKeyName('adminlist'), adminlist),\n      initSet(this.redis, this.makeKeyName('userlist'), null),\n      this.redis.del(this.makeKeyName('messagesHistory')),\n      this.redis.del(this.makeKeyName('messagesTimestamps')),\n      this.redis.del(this.makeKeyName('messagesIds')),\n      this.redis.del(this.makeKeyName('usersseen')),\n      this.redis.set(this.makeKeyName('lastMessageId'), 0),\n      this.redis.set(this.makeKeyName('owner'), owner),\n      this.whitelistOnlySet(whitelistOnly),\n      this.accessListsUpdatesSet(enableAccessListsUpdates),\n      this.userlistUpdatesSet(enableUserlistUpdates),\n      this.historyMaxSizeSet(historyMaxSize)\n    ]).return()\n  }\n\n  hasList (listName) {\n    return listName === 'adminlist' || listName === 'whitelist' ||\n      listName === 'blacklist' || listName === 'userlist'\n  }\n\n  ownerGet () {\n    return this.redis.get(this.makeKeyName('owner'))\n  }\n\n  ownerSet (owner) {\n    return this.redis.set(this.makeKeyName('owner'), owner)\n  }\n\n  accessListsUpdatesSet (enableAccessListsUpdates) {\n    enableAccessListsUpdates = enableAccessListsUpdates ? true : ''\n    return this.redis.set(this.makeKeyName('enableAccessListsUpdates'),\n                          enableAccessListsUpdates)\n  }\n\n  accessListsUpdatesGet () {\n    return this.redis.get(this.makeKeyName('enableAccessListsUpdates'))\n      .then(data => Promise.resolve(Boolean(data)))\n  }\n\n  userlistUpdatesSet (enableUserlistUpdates) {\n    enableUserlistUpdates = enableUserlistUpdates ? true : ''\n    return this.redis.set(this.makeKeyName('enableUserlistUpdates'),\n                          enableUserlistUpdates)\n  }\n\n  userlistUpdatesGet () {\n    return this.redis.get(this.makeKeyName('enableUserlistUpdates'))\n      .then(data => Promise.resolve(Boolean(data)))\n  }\n\n  historyMaxSizeSet (historyMaxSize) {\n    let limit = historyMaxSize\n    if (!(_.isNumber(historyMaxSize) && historyMaxSize >= 0)) {\n      limit = this.server.historyMaxSize\n    }\n    if (limit === 0) {\n      return this.redis.multi()\n        .set(this.makeKeyName('historyMaxSize'), limit)\n        .del(this.makeKeyName('messagesHistory'))\n        .del(this.makeKeyName('messagesTimestamps'))\n        .del(this.makeKeyName('messagesIds'))\n        .exec()\n    } else {\n      let last = limit - 1\n      return this.redis.multi()\n        .set(this.makeKeyName('historyMaxSize'), limit)\n        .ltrim(this.makeKeyName('messagesHistory'), 0, last)\n        .ltrim(this.makeKeyName('messagesTimestamps'), 0, last)\n        .ltrim(this.makeKeyName('messagesIds'), 0, last)\n        .exec()\n    }\n  }\n\n  historyInfo () {\n    return this.redis.multi()\n      .get(this.makeKeyName('historyMaxSize'))\n      .llen(this.makeKeyName('messagesHistory'))\n      .get(this.makeKeyName('lastMessageId'))\n      .exec()\n      .spread(([, historyMaxSize], [, historySize], [, lastMessageId]) => {\n        historySize = parseInt(historySize)\n        historyMaxSize = parseFloat(historyMaxSize)\n        lastMessageId = parseInt(lastMessageId)\n        let info = { historySize,\n                     historyMaxSize,\n                     historyMaxGetMessages: this.historyMaxGetMessages,\n                     lastMessageId }\n        return Promise.resolve(info)\n      })\n  }\n\n  getCommonUsers () {\n    return this.redis.sdiff(this.makeKeyName('userlist'),\n                            this.makeKeyName('whitelist'),\n                            this.makeKeyName('adminlist'))\n  }\n\n  messageAdd (msg) {\n    let timestamp = _.now()\n    let smsg = JSON.stringify(msg)\n    return this.redis.messageAdd(\n      this.makeKeyName('lastMessageId'), this.makeKeyName('historyMaxSize'),\n      this.makeKeyName('messagesIds'), this.makeKeyName('messagesTimestamps'),\n      this.makeKeyName('messagesHistory'), smsg, timestamp)\n      .spread(id => {\n        msg.id = id\n        msg.timestamp = timestamp\n        return Promise.resolve(msg)\n      })\n  }\n\n  convertMessages (msgs, tss, ids) {\n    let data = []\n    if (!msgs) {\n      return Promise.resolve(data)\n    }\n    for (let idx = 0; idx < msgs.length; idx++) {\n      let msg = msgs[idx]\n      let obj = JSON.parse(msg, (key, val) => {\n        if (val && val.type === 'Buffer') {\n          return new Buffer(val.data)\n        } else {\n          return val\n        }\n      })\n      obj.timestamp = parseInt(tss[idx])\n      obj.id = parseInt(ids[idx])\n      data[idx] = obj\n    }\n    return Promise.resolve(data)\n  }\n\n  messagesGetRecent () {\n    if (this.historyMaxGetMessages <= 0) { return Promise.resolve([]) }\n    let limit = this.historyMaxGetMessages - 1\n    return this.redis.multi()\n      .lrange(this.makeKeyName('messagesHistory'), 0, limit)\n      .lrange(this.makeKeyName('messagesTimestamps'), 0, limit)\n      .lrange(this.makeKeyName('messagesIds'), 0, limit)\n      .exec()\n      .spread(([, msgs], [, tss], [, ids]) => {\n        return this.convertMessages(msgs, tss, ids)\n      })\n  }\n\n  messagesGet (id, maxMessages = this.historyMaxGetMessages) {\n    if (maxMessages <= 0) { return Promise.resolve([]) }\n    id = _.max([0, id])\n    return this.redis.messagesGet(\n      this.makeKeyName('lastMessageId'), this.makeKeyName('historyMaxSize'),\n      this.makeKeyName('messagesIds'), this.makeKeyName('messagesTimestamps'),\n      this.makeKeyName('messagesHistory'), id, maxMessages)\n      .spread((msgs, tss, ids) => {\n        return this.convertMessages(msgs, tss, ids)\n      })\n  }\n\n  userSeenGet (userName) {\n    return this.redis.multi()\n      .hget(this.makeKeyName('usersseen'), userName)\n      .sismember(this.makeKeyName('userlist'), userName)\n      .exec()\n      .spread(([, ts], [, isjoined]) => {\n        let joined = Boolean(isjoined)\n        let timestamp = ts ? parseInt(ts) : null\n        return {joined, timestamp}\n      })\n  }\n\n  userSeenUpdate (userName) {\n    let timestamp = _.now()\n    return this.redis.hset(this.makeKeyName('usersseen'), userName, timestamp)\n  }\n\n}\n\n// Implements direct messaging state API.\nclass DirectMessagingStateRedis extends ListsStateRedis {\n\n  constructor (server, userName) {\n    super()\n    this.server = server\n    this.userName = userName\n    this.name = this.userName\n    this.prefix = 'users'\n    this.exitsErrorName = 'userExists'\n    this.redis = this.server.redis\n    mixin(this, StateOperations, this.name, this.exitsErrorName, this.redis,\n          this.makeKeyName.bind(this), this.stateReset.bind(this))\n  }\n\n  hasList (listName) {\n    return listName === 'whitelist' || listName === 'blacklist'\n  }\n\n  stateReset (state) {\n    state = state || {}\n    let { whitelist, blacklist, whitelistOnly } = state\n    whitelistOnly = whitelistOnly ? true : ''\n    return Promise.all([\n      initSet(this.redis, this.makeKeyName('whitelist'), whitelist),\n      initSet(this.redis, this.makeKeyName('blacklist'), blacklist),\n      this.redis.set(this.makeKeyName('whitelistMode'), whitelistOnly)\n    ]).return()\n  }\n\n}\n\n// Implements user state API.\nclass UserStateRedis {\n\n  constructor (server, userName) {\n    this.server = server\n    this.userName = userName\n    this.name = this.userName\n    this.prefix = 'users'\n    this.redis = this.server.redis\n    mixin(this, LockOperations, this.redis)\n  }\n\n  makeKeyName (keyName) {\n    return `${namespace}:${this.prefix}:{${this.name}}:${keyName}`\n  }\n\n  makeSocketToRooms (id = '') {\n    return this.makeKeyName(`socketsToRooms:${id}`)\n  }\n\n  makeRoomToSockets (room = '') {\n    return this.makeKeyName(`roomsToSockets:${room}`)\n  }\n\n  makeRoomLock (room) {\n    return this.makeKeyName(`roomLock:${room}`)\n  }\n\n  addSocket (id, uid) {\n    return this.redis.multi()\n      .hset(this.makeKeyName('sockets'), id, uid)\n      .hlen(this.makeKeyName('sockets'))\n      .exec()\n      .spread((_, [, nconnected]) => Promise.resolve(nconnected))\n  }\n\n  getAllSockets () {\n    return this.redis.hkeys(this.makeKeyName('sockets'))\n  }\n\n  getSocketsToInstance () {\n    return this.redis.hgetall(this.makeKeyName('sockets'))\n  }\n\n  getRoomToSockets (roomName) {\n    return this.redis.smembers(this.makeRoomToSockets(roomName))\n  }\n\n  getSocketsToRooms () {\n    return this.redis.getSocketsToRooms(\n      this.makeKeyName('sockets'), this.makeSocketToRooms())\n      .spread(result => {\n        let data = JSON.parse(result) || {}\n        for (let [k, v] of _.toPairs(data)) {\n          if (_.isEmpty(v)) { data[k] = [] }\n        }\n        return Promise.resolve(data)\n      })\n  }\n\n  addSocketToRoom (id, roomName) {\n    return this.redis.multi()\n      .sadd(this.makeSocketToRooms(id), roomName)\n      .sadd(this.makeRoomToSockets(roomName), id)\n      .scard(this.makeRoomToSockets(roomName))\n      .exec()\n      .then(([, , [, njoined]]) => Promise.resolve(njoined))\n  }\n\n  removeSocketFromRoom (id, roomName) {\n    return this.redis.multi()\n      .scard(this.makeRoomToSockets(roomName))\n      .srem(this.makeSocketToRooms(id), roomName)\n      .srem(this.makeRoomToSockets(roomName), id)\n      .scard(this.makeRoomToSockets(roomName))\n      .exec()\n      .then(([[, wasjoined], , , [, njoined]]) => {\n        let hasChanged = njoined !== wasjoined\n        return Promise.resolve([njoined, hasChanged])\n      })\n  }\n\n  removeAllSocketsFromRoom (roomName) {\n    return this.redis.removeAllSocketsFromRoom(\n      this.makeRoomToSockets(roomName), this.makeSocketToRooms(), roomName)\n      .spread(result => Promise.resolve(JSON.parse(result)))\n  }\n\n  removeSocket (id) {\n    return this.redis.removeSocket(\n      this.makeSocketToRooms(id), this.makeKeyName('sockets'),\n      this.makeRoomToSockets(), id)\n      .spread(result => Promise.resolve(JSON.parse(result)))\n  }\n\n  lockToRoom (roomName, ttl) {\n    return uid(18).then(val => {\n      let start = _.now()\n      return this.lock(this.makeRoomLock(roomName), val, ttl).then(() => {\n        return Promise.resolve().disposer(() => {\n          if (start + ttl < _.now()) {\n            this.server.emit(\n              'lockTimeExceeded', val, {userName: this.userName, roomName})\n          }\n          return this.unlock(this.makeRoomLock(roomName), val)\n        })\n      })\n    })\n  }\n\n}\n\n// Implements global state API.\nclass RedisState {\n\n  constructor (server, options) {\n    this.server = server\n    this.options = options\n    this.closed = false\n    if (this.options.useCluster) {\n      this.redis = new Redis.Cluster(...this.options.redisOptions)\n    } else {\n      let redisOptions = _.castArray(this.options.redisOptions)\n      this.redis = new Redis(...redisOptions)\n    }\n    this.RoomState = RoomStateRedis\n    this.UserState = UserStateRedis\n    this.DirectMessagingState = DirectMessagingStateRedis\n    this.lockTTL = this.options.lockTTL || 10000\n    this.instanceUID = this.server.instanceUID\n    this.server.redis = this.redis\n    for (let [cmd, def] of _.toPairs(luaCommands)) {\n      this.redis.defineCommand(cmd, {\n        numberOfKeys: def.numberOfKeys,\n        lua: def.lua\n      })\n    }\n  }\n\n  makeKeyName (prefix, name, keyName) {\n    return `${namespace}:${prefix}:{${name}}:${keyName}`\n  }\n\n  hasRoom (name) {\n    return this.redis.get(this.makeKeyName('rooms', name, 'isInit'))\n  }\n\n  hasUser (name) {\n    return this.redis.get(this.makeKeyName('users', name, 'isInit'))\n  }\n\n  close () {\n    this.closed = true\n    return this.redis.quit().return()\n  }\n\n  getRoom (name, isPredicate = false) {\n    let room = new Room(this.server, name)\n    return this.hasRoom(name).then(exists => {\n      if (!exists) {\n        if (isPredicate) {\n          return Promise.resolve(null)\n        } else {\n          let error = new ChatServiceError('noRoom', name)\n          return Promise.reject(error)\n        }\n      }\n      return Promise.resolve(room)\n    })\n  }\n\n  addRoom (name, state) {\n    let room = new Room(this.server, name)\n    return room.initState(state).return(room)\n  }\n\n  removeRoom (name) {\n    return Promise.resolve()\n  }\n\n  addSocket (id, userName) {\n    return this.redis.hset(\n      this.makeKeyName('instances', this.instanceUID, 'sockets'), id, userName)\n  }\n\n  removeSocket (id) {\n    return this.redis.hdel(\n      this.makeKeyName('instances', this.instanceUID, 'sockets'), id)\n  }\n\n  getInstanceSockets (uid = this.instanceUID) {\n    return this.redis.hgetall(this.makeKeyName('instances', uid, 'sockets'))\n  }\n\n  updateHeartbeat () {\n    return this.redis.set(\n      this.makeKeyName('instances', this.instanceUID, 'heartbeat'), _.now())\n      .catchReturn()\n  }\n\n  getInstanceHeartbeat (uid = this.instanceUID) {\n    return this.redis.get(this.makeKeyName('instances', uid, 'heartbeat'))\n      .then(ts => ts ? parseInt(ts) : null)\n  }\n\n  getOrAddUser (name, state) {\n    let user = new User(this.server, name)\n    return this.hasUser(name).then(exists => {\n      if (!exists) {\n        return user.initState(state)\n      } else {\n        return Promise.resolve()\n      }\n    }).catch(ChatServiceError, e => user)\n      .return(user)\n  }\n\n  getUser (name, isPredicate = false) {\n    let user = new User(this.server, name)\n    return this.hasUser(name).then(exists => {\n      if (!exists) {\n        if (isPredicate) {\n          return Promise.resolve(null)\n        } else {\n          let error = new ChatServiceError('noUser', name)\n          return Promise.reject(error)\n        }\n      }\n      return Promise.resolve(user)\n    })\n  }\n\n  addUser (name, state) {\n    let user = new User(this.server, name)\n    return user.initState(state).return(user)\n  }\n\n  removeUser (name) {\n    return Promise.resolve()\n  }\n\n}\n\nmodule.exports = RedisState\n"]} |
\ | No newline at end of file |