UNPKG

87.8 kBJavaScriptView Raw
1'use strict';
2
3var _toConsumableArray2 = require('babel-runtime/helpers/toConsumableArray');
4
5var _toConsumableArray3 = _interopRequireDefault(_toConsumableArray2);
6
7var _getIterator2 = require('babel-runtime/core-js/get-iterator');
8
9var _getIterator3 = _interopRequireDefault(_getIterator2);
10
11var _stringify = require('babel-runtime/core-js/json/stringify');
12
13var _stringify2 = _interopRequireDefault(_stringify);
14
15var _slicedToArray2 = require('babel-runtime/helpers/slicedToArray');
16
17var _slicedToArray3 = _interopRequireDefault(_slicedToArray2);
18
19var _getPrototypeOf = require('babel-runtime/core-js/object/get-prototype-of');
20
21var _getPrototypeOf2 = _interopRequireDefault(_getPrototypeOf);
22
23var _possibleConstructorReturn2 = require('babel-runtime/helpers/possibleConstructorReturn');
24
25var _possibleConstructorReturn3 = _interopRequireDefault(_possibleConstructorReturn2);
26
27var _inherits2 = require('babel-runtime/helpers/inherits');
28
29var _inherits3 = _interopRequireDefault(_inherits2);
30
31var _classCallCheck2 = require('babel-runtime/helpers/classCallCheck');
32
33var _classCallCheck3 = _interopRequireDefault(_classCallCheck2);
34
35var _createClass2 = require('babel-runtime/helpers/createClass');
36
37var _createClass3 = _interopRequireDefault(_createClass2);
38
39function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
40
41var ChatServiceError = require('./ChatServiceError');
42var Promise = require('bluebird');
43var Redis = require('ioredis');
44var Room = require('./Room');
45var User = require('./User');
46var _ = require('lodash');
47var promiseRetry = require('promise-retry');
48var uid = require('uid-safe');
49
50var _require = require('es6-mixin');
51
52var mixin = _require.mixin;
53
54
55var namespace = 'chatservice';
56
57function 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
69var 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
119var 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
154var 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
189var 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
277var 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
508var 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
550var 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
717var 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
896module.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