UNPKG

32.9 kBJavaScriptView Raw
1'use strict';
2
3var _slicedToArray2 = require('babel-runtime/helpers/slicedToArray');
4
5var _slicedToArray3 = _interopRequireDefault(_slicedToArray2);
6
7var _classCallCheck2 = require('babel-runtime/helpers/classCallCheck');
8
9var _classCallCheck3 = _interopRequireDefault(_classCallCheck2);
10
11var _createClass2 = require('babel-runtime/helpers/createClass');
12
13var _createClass3 = _interopRequireDefault(_createClass2);
14
15function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
16
17var ChatServiceError = require('./ChatServiceError');
18var Promise = require('bluebird');
19var _ = require('lodash');
20
21var _require = require('./utils');
22
23var checkNameSymbols = _require.checkNameSymbols;
24var possiblyCallback = _require.possiblyCallback;
25
26/**
27 * Server side operations.
28 *
29 * @mixin
30 * @memberof chat-service
31 * @see chat-service.ChatService
32 */
33
34var ServiceAPI = function () {
35 function ServiceAPI(state, makeUser, clusterBus) {
36 (0, _classCallCheck3.default)(this, ServiceAPI);
37
38 this.state = state;
39 this.makeUser = makeUser;
40 this.clusterBus = clusterBus;
41 }
42
43 /**
44 * Executes {@link rpc.clientRequests} handlers.
45 *
46 * @param {string|boolean|Object} context Is a `userName` if
47 * `string`, or a `bypassPermissions` if `boolean`, or an options
48 * object if `Object`.
49 * @param {string} command Command name.
50 * @param {...*} args Command arguments.
51 * @param {callback} [cb] Optional callback.
52 *
53 * @property {string} [context.userName] User name.
54 * @property {string} [context.id] Socket id, it is required for
55 * {@link rpc.clientRequests.roomJoin} and {@link
56 * rpc.clientRequests.roomLeave} commands.
57 * @property {boolean} [context.bypassHooks=false] If `false`
58 * executes command without before and after hooks.
59 * @property {boolean} [context.bypassPermissions=false] If `true`
60 * executes command (except {@link rpc.clientRequests.roomJoin})
61 * bypassing built-in permissions checking.
62 *
63 * @return {Promise<Array>} Array of command results.
64 *
65 * @see rpc.clientRequests
66 */
67
68
69 (0, _createClass3.default)(ServiceAPI, [{
70 key: 'execUserCommand',
71 value: function execUserCommand(context, command) {
72 var _this = this;
73
74 if (_.isObject(context)) {
75 var _context = context;
76 var userName = _context.userName;
77
78 context = _.clone(context);
79 } else if (_.isBoolean(context)) {
80 context = { bypassPermissions: context };
81 } else {
82 userName = context;
83 context = {};
84 }
85 context.isLocalCall = true;
86
87 for (var _len = arguments.length, args = Array(_len > 2 ? _len - 2 : 0), _key = 2; _key < _len; _key++) {
88 args[_key - 2] = arguments[_key];
89 }
90
91 var _possiblyCallback = possiblyCallback(args);
92
93 var _possiblyCallback2 = (0, _slicedToArray3.default)(_possiblyCallback, 2);
94
95 var nargs = _possiblyCallback2[0];
96 var cb = _possiblyCallback2[1];
97
98 return Promise.try(function () {
99 if (userName) {
100 return _this.state.getUser(userName);
101 } else {
102 return _this.makeUser();
103 }
104 }).then(function (user) {
105 return user.exec(command, context, nargs);
106 }).asCallback(cb, { spread: true });
107 }
108
109 /**
110 * Adds an user with a state.
111 *
112 * @param {string} userName User name.
113 * @param {Object} [state] User state.
114 * @param {callback} [cb] Optional callback.
115 *
116 * @property {Array<string>} [state.whitelist=[]] User direct messages
117 * whitelist.
118 * @property {Array<string>} [state.blacklist=[]] User direct messages
119 * blacklist.
120 * @property {boolean} [state.whitelistOnly=false] User direct
121 * messages whitelistOnly mode.
122 *
123 * @return {Promise<undefined>} Promise that resolves without any data.
124 */
125
126 }, {
127 key: 'addUser',
128 value: function addUser(userName, state, cb) {
129 var _this2 = this;
130
131 return checkNameSymbols(userName).then(function () {
132 return _this2.state.addUser(userName, state);
133 }).return().asCallback(cb);
134 }
135
136 /**
137 * Deletes an offline user. Will raise an error if a user has online
138 * sockets.
139 *
140 * @param {string} userName User name.
141 * @param {callback} [cb] Optional callback.
142 *
143 * @return {Promise<undefined>} Promise that resolves without any data.
144 */
145
146 }, {
147 key: 'deleteUser',
148 value: function deleteUser(userName, cb) {
149 var _this3 = this;
150
151 return this.state.getUser(userName).then(function (user) {
152 return user.listOwnSockets().then(function (sockets) {
153 if (sockets && _.size(sockets) > 0) {
154 return Promise.reject(new ChatServiceError('userOnline', userName));
155 } else {
156 return Promise.all([user.removeState(), _this3.state.removeUser(userName)]);
157 }
158 });
159 }).return().asCallback(cb);
160 }
161
162 /**
163 * Checks for an user existence.
164 *
165 * @param {string} userName User name.
166 * @param {callback} [cb] Optional callback.
167 *
168 * @return {Promise<boolean>} Predicate result.
169 */
170
171 }, {
172 key: 'hasUser',
173 value: function hasUser(userName, cb) {
174 return this.state.getUser(userName, true).then(function (user) {
175 return Boolean(user);
176 }).asCallback(cb);
177 }
178
179 /**
180 * Checks for a name existence in a direct messaging list.
181 *
182 * @param {string} userName User name.
183 * @param {string} listName List name.
184 * @param {string} item List element.
185 * @param {callback} [cb] Optional callback.
186 *
187 * @return {Promise<boolean>} Predicate result.
188 */
189
190 }, {
191 key: 'userHasInList',
192 value: function userHasInList(userName, listName, item, cb) {
193 return this.state.getUser(userName).then(function (user) {
194 return user.directMessaging.hasInList(listName, item);
195 }).asCallback(cb);
196 }
197
198 /**
199 * Checks for a direct messaging permission.
200 *
201 * @param {string} recipient Recipient name.
202 * @param {string} sender Sender name.
203 * @param {callback} [cb] Optional callback.
204 *
205 * @return {Promise<boolean>} Predicate result.
206 */
207
208 }, {
209 key: 'hasDirectAccess',
210 value: function hasDirectAccess(recipient, sender, cb) {
211 return this.state.getUser(recipient).then(function (user) {
212 return user.directMessaging.checkAcess(sender);
213 }).return(true).catchReturn(ChatServiceError, false).asCallback(cb);
214 }
215
216 /**
217 * Disconnects user's sockets for all service instances. Method is
218 * asynchronous, returns without waiting for the completion.
219 *
220 * @param {string} userName User name.
221 *
222 * @return {undefined} Returns no data.
223 */
224
225 }, {
226 key: 'disconnectUserSockets',
227 value: function disconnectUserSockets(userName) {
228 this.clusterBus.emit('disconnectUserSockets', userName);
229 }
230
231 /**
232 * Adds a room with a state.
233 *
234 * @param {string} roomName Room name.
235 * @param {Object} [state] Room state.
236 * @param {callback} [cb] Optional callback.
237 *
238 * @property {Array<string>} [state.whitelist=[]] Room whitelist.
239 * @property {Array<string>} [state.blacklist=[]] Room blacklist
240 * @property {Array<string>} [state.adminlist=[]] Room adminlist.
241 * @property {boolean} [state.whitelistOnly=false] Room
242 * whitelistOnly mode.
243 * @property {string} [state.owner] Room owner.
244 * @property {number} [state.historyMaxSize] Room history maximum
245 * size. Defalut value is {@link chat-service.config.options}
246 * `historyMaxSize`.
247 * @property {boolean} [state.enableAccessListsUpdates] Room enable
248 * access lists updates. Defalut value is {@link
249 * chat-service.config.options} `enableAccessListsUpdates`.
250 * @property {boolean} [state.enableUserlistUpdates] Room enable
251 * userlist updates. Defalut value is {@link
252 * chat-service.config.options} `enableUserlistUpdates`.
253 *
254 * @return {Promise<undefined>} Promise that resolves without any data.
255 */
256
257 }, {
258 key: 'addRoom',
259 value: function addRoom(roomName, state, cb) {
260 var _this4 = this;
261
262 return checkNameSymbols(roomName).then(function () {
263 return _this4.state.addRoom(roomName, state);
264 }).return().asCallback(cb);
265 }
266
267 /**
268 * Removes all joined users from the room and removes all room data.
269 *
270 * @param {string} roomName Room name.
271 * @param {callback} [cb] Optional callback.
272 *
273 * @return {Promise<undefined>} Promise that resolves without any data.
274 */
275
276 }, {
277 key: 'deleteRoom',
278 value: function deleteRoom(roomName, cb) {
279 return this.execUserCommand(true, 'roomDelete', roomName).return().asCallback(cb);
280 }
281
282 /**
283 * Checks for a room existence.
284 *
285 * @param {string} roomName Room name.
286 * @param {callback} [cb] Optional callback.
287 *
288 * @return {Promise<boolean>} Predicate result.
289 */
290
291 }, {
292 key: 'hasRoom',
293 value: function hasRoom(roomName, cb) {
294 return this.state.getRoom(roomName, true).then(function (room) {
295 return Boolean(room);
296 }).asCallback(cb);
297 }
298
299 /**
300 * Checks for a name existence in a room list.
301 *
302 * @param {string} roomName Room name.
303 * @param {string} listName List name.
304 * @param {string} item List element.
305 * @param {callback} [cb] Optional callback.
306 *
307 * @return {Promise<boolean>} Predicate result.
308 */
309
310 }, {
311 key: 'roomHasInList',
312 value: function roomHasInList(roomName, listName, item, cb) {
313 return this.state.getRoom(roomName).then(function (room) {
314 return room.roomState.hasInList(listName, item);
315 }).asCallback(cb);
316 }
317
318 /**
319 * Checks for a room access permission.
320 *
321 * @param {string} roomName Room name.
322 * @param {string} userName User name.
323 * @param {callback} [cb] Optional callback.
324 *
325 * @return {Promise<boolean>} Predicate result.
326 */
327
328 }, {
329 key: 'hasRoomAccess',
330 value: function hasRoomAccess(roomName, userName, cb) {
331 return this.state.getRoom(roomName).then(function (room) {
332 return room.checkAcess(userName);
333 }).return(true).catchReturn(ChatServiceError, false).asCallback(cb);
334 }
335
336 /**
337 * Changes the room owner.
338 *
339 * @param {string} roomName Room name.
340 * @param {string} owner Owner user name.
341 * @param {callback} [cb] Optional callback.
342 *
343 * @return {Promise<undefined>} Promise that resolves without any data.
344 */
345
346 }, {
347 key: 'changeRoomOwner',
348 value: function changeRoomOwner(roomName, owner, cb) {
349 return this.state.getRoom(roomName).then(function (room) {
350 return room.roomState.ownerSet(owner);
351 }).return().asCallback(cb);
352 }
353
354 /**
355 * Changes the room history size.
356 *
357 * @param {string} roomName Room name.
358 * @param {number} size Room history size.
359 * @param {callback} [cb] Optional callback.
360 *
361 * @return {Promise<undefined>} Promise that resolves without any data.
362 */
363
364 }, {
365 key: 'changeRoomHistoryMaxSize',
366 value: function changeRoomHistoryMaxSize(roomName, size, cb) {
367 return this.state.getRoom(roomName).then(function (room) {
368 return room.roomState.historyMaxSizeSet(size);
369 }).return().asCallback(cb);
370 }
371
372 /**
373 * Enables or disables access lists updates for the room.
374 *
375 * @param {string} roomName Room name.
376 * @param {boolean} mode Enable or disable.
377 * @param {callback} [cb] Optional callback.
378 *
379 * @return {Promise<undefined>} Promise that resolves without any data.
380 *
381 * @see rpc.serverNotifications.roomAccessListAdded
382 * @see rpc.serverNotifications.roomAccessListRemoved
383 * @see rpc.serverNotifications.roomModeChanged
384 */
385
386 }, {
387 key: 'changeAccessListsUpdates',
388 value: function changeAccessListsUpdates(roomName, mode, cb) {
389 return this.state.getRoom(roomName).then(function (room) {
390 return room.roomState.accessListsUpdatesSet(mode);
391 }).return().asCallback(cb);
392 }
393
394 /**
395 * Enables or disables user list updates for the room.
396 *
397 * @param {string} roomName Room name.
398 * @param {boolean} mode Enable or disable.
399 * @param {callback} [cb] Optional callback.
400 *
401 * @return {Promise<undefined>} Promise that resolves without any data.
402 *
403 * @see rpc.serverNotifications.roomUserJoined
404 * @see rpc.serverNotifications.roomUserLeft
405 */
406
407 }, {
408 key: 'changeUserlistUpdates',
409 value: function changeUserlistUpdates(roomName, mode, cb) {
410 return this.state.getRoom(roomName).then(function (room) {
411 return room.roomState.userlistUpdatesSet(mode);
412 }).return().asCallback(cb);
413 }
414 }]);
415 return ServiceAPI;
416}();
417
418module.exports = ServiceAPI;
419//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/ServiceAPI.js"],"names":[],"mappings":"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","file":"ServiceAPI.js","sourcesContent":["'use strict'\n\nconst ChatServiceError = require('./ChatServiceError')\nconst Promise = require('bluebird')\nconst _ = require('lodash')\nconst { checkNameSymbols, possiblyCallback } = require('./utils')\n\n/**\n * Server side operations.\n *\n * @mixin\n * @memberof chat-service\n * @see chat-service.ChatService\n */\nclass ServiceAPI {\n\n  constructor (state, makeUser, clusterBus) {\n    this.state = state\n    this.makeUser = makeUser\n    this.clusterBus = clusterBus\n  }\n\n  /**\n   * Executes {@link rpc.clientRequests} handlers.\n   *\n   * @param {string|boolean|Object} context Is a `userName` if\n   * `string`, or a `bypassPermissions` if `boolean`, or an options\n   * object if `Object`.\n   * @param {string} command Command name.\n   * @param {...*} args Command arguments.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @property {string} [context.userName] User name.\n   * @property {string} [context.id] Socket id, it is required for\n   * {@link rpc.clientRequests.roomJoin} and {@link\n   * rpc.clientRequests.roomLeave} commands.\n   * @property {boolean} [context.bypassHooks=false] If `false`\n   * executes command without before and after hooks.\n   * @property {boolean} [context.bypassPermissions=false] If `true`\n   * executes command (except {@link rpc.clientRequests.roomJoin})\n   * bypassing built-in permissions checking.\n   *\n   * @return {Promise<Array>} Array of command results.\n   *\n   * @see rpc.clientRequests\n   */\n  execUserCommand (context, command, ...args) {\n    if (_.isObject(context)) {\n      var { userName } = context\n      context = _.clone(context)\n    } else if (_.isBoolean(context)) {\n      context = {bypassPermissions: context}\n    } else {\n      userName = context\n      context = {}\n    }\n    context.isLocalCall = true\n    let [nargs, cb] = possiblyCallback(args)\n    return Promise.try(() => {\n      if (userName) {\n        return this.state.getUser(userName)\n      } else {\n        return this.makeUser()\n      }\n    }).then(user => user.exec(command, context, nargs))\n      .asCallback(cb, { spread: true })\n  }\n\n  /**\n   * Adds an user with a state.\n   *\n   * @param {string} userName User name.\n   * @param {Object} [state] User state.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @property {Array<string>} [state.whitelist=[]] User direct messages\n   * whitelist.\n   * @property {Array<string>} [state.blacklist=[]] User direct messages\n   * blacklist.\n   * @property {boolean} [state.whitelistOnly=false] User direct\n   * messages whitelistOnly mode.\n   *\n   * @return {Promise<undefined>} Promise that resolves without any data.\n   */\n  addUser (userName, state, cb) {\n    return checkNameSymbols(userName)\n      .then(() => this.state.addUser(userName, state))\n      .return()\n      .asCallback(cb)\n  }\n\n  /**\n   * Deletes an offline user. Will raise an error if a user has online\n   * sockets.\n   *\n   * @param {string} userName User name.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @return {Promise<undefined>} Promise that resolves without any data.\n   */\n  deleteUser (userName, cb) {\n    return this.state.getUser(userName).then(user => {\n      return user.listOwnSockets().then(sockets => {\n        if (sockets && _.size(sockets) > 0) {\n          return Promise.reject(new ChatServiceError('userOnline', userName))\n        } else {\n          return Promise.all([\n            user.removeState(),\n            this.state.removeUser(userName)\n          ])\n        }\n      })\n    }).return().asCallback(cb)\n  }\n\n  /**\n   * Checks for an user existence.\n   *\n   * @param {string} userName User name.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @return {Promise<boolean>} Predicate result.\n   */\n  hasUser (userName, cb) {\n    return this.state.getUser(userName, true)\n      .then(user => Boolean(user))\n      .asCallback(cb)\n  }\n\n  /**\n   * Checks for a name existence in a direct messaging list.\n   *\n   * @param {string} userName User name.\n   * @param {string} listName List name.\n   * @param {string} item List element.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @return {Promise<boolean>} Predicate result.\n   */\n  userHasInList (userName, listName, item, cb) {\n    return this.state.getUser(userName)\n      .then(user => user.directMessaging.hasInList(listName, item))\n      .asCallback(cb)\n  }\n\n  /**\n   * Checks for a direct messaging permission.\n   *\n   * @param {string} recipient Recipient name.\n   * @param {string} sender Sender name.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @return {Promise<boolean>} Predicate result.\n   */\n  hasDirectAccess (recipient, sender, cb) {\n    return this.state.getUser(recipient)\n      .then(user => user.directMessaging.checkAcess(sender))\n      .return(true)\n      .catchReturn(ChatServiceError, false)\n      .asCallback(cb)\n  }\n\n  /**\n   * Disconnects user's sockets for all service instances. Method is\n   * asynchronous, returns without waiting for the completion.\n   *\n   * @param {string} userName User name.\n   *\n   * @return {undefined} Returns no data.\n   */\n  disconnectUserSockets (userName) {\n    this.clusterBus.emit('disconnectUserSockets', userName)\n  }\n\n  /**\n   * Adds a room with a state.\n   *\n   * @param {string} roomName Room name.\n   * @param {Object} [state] Room state.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @property {Array<string>} [state.whitelist=[]] Room whitelist.\n   * @property {Array<string>} [state.blacklist=[]] Room blacklist\n   * @property {Array<string>} [state.adminlist=[]] Room adminlist.\n   * @property {boolean} [state.whitelistOnly=false] Room\n   * whitelistOnly mode.\n   * @property {string} [state.owner] Room owner.\n   * @property {number} [state.historyMaxSize] Room history maximum\n   * size. Defalut value is {@link chat-service.config.options}\n   * `historyMaxSize`.\n   * @property {boolean} [state.enableAccessListsUpdates] Room enable\n   * access lists updates. Defalut value is {@link\n   * chat-service.config.options} `enableAccessListsUpdates`.\n   * @property {boolean} [state.enableUserlistUpdates] Room enable\n   * userlist updates. Defalut value is {@link\n   * chat-service.config.options} `enableUserlistUpdates`.\n   *\n   * @return {Promise<undefined>} Promise that resolves without any data.\n   */\n  addRoom (roomName, state, cb) {\n    return checkNameSymbols(roomName)\n      .then(() => this.state.addRoom(roomName, state))\n      .return()\n      .asCallback(cb)\n  }\n\n  /**\n   * Removes all joined users from the room and removes all room data.\n   *\n   * @param {string} roomName Room name.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @return {Promise<undefined>} Promise that resolves without any data.\n   */\n  deleteRoom (roomName, cb) {\n    return this.execUserCommand(true, 'roomDelete', roomName)\n      .return()\n      .asCallback(cb)\n  }\n\n  /**\n   * Checks for a room existence.\n   *\n   * @param {string} roomName Room name.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @return {Promise<boolean>} Predicate result.\n   */\n  hasRoom (roomName, cb) {\n    return this.state.getRoom(roomName, true)\n      .then(room => Boolean(room))\n      .asCallback(cb)\n  }\n\n  /**\n   * Checks for a name existence in a room list.\n   *\n   * @param {string} roomName Room name.\n   * @param {string} listName List name.\n   * @param {string} item List element.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @return {Promise<boolean>} Predicate result.\n   */\n  roomHasInList (roomName, listName, item, cb) {\n    return this.state.getRoom(roomName)\n      .then(room => room.roomState.hasInList(listName, item))\n      .asCallback(cb)\n  }\n\n  /**\n   * Checks for a room access permission.\n   *\n   * @param {string} roomName Room name.\n   * @param {string} userName User name.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @return {Promise<boolean>} Predicate result.\n   */\n  hasRoomAccess (roomName, userName, cb) {\n    return this.state.getRoom(roomName)\n      .then(room => room.checkAcess(userName))\n      .return(true)\n      .catchReturn(ChatServiceError, false)\n      .asCallback(cb)\n  }\n\n  /**\n   * Changes the room owner.\n   *\n   * @param {string} roomName Room name.\n   * @param {string} owner Owner user name.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @return {Promise<undefined>} Promise that resolves without any data.\n   */\n  changeRoomOwner (roomName, owner, cb) {\n    return this.state.getRoom(roomName)\n      .then(room => room.roomState.ownerSet(owner))\n      .return()\n      .asCallback(cb)\n  }\n\n  /**\n   * Changes the room history size.\n   *\n   * @param {string} roomName Room name.\n   * @param {number} size Room history size.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @return {Promise<undefined>} Promise that resolves without any data.\n   */\n  changeRoomHistoryMaxSize (roomName, size, cb) {\n    return this.state.getRoom(roomName)\n      .then(room => room.roomState.historyMaxSizeSet(size))\n      .return()\n      .asCallback(cb)\n  }\n\n  /**\n   * Enables or disables access lists updates for the room.\n   *\n   * @param {string} roomName Room name.\n   * @param {boolean} mode Enable or disable.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @return {Promise<undefined>} Promise that resolves without any data.\n   *\n   * @see rpc.serverNotifications.roomAccessListAdded\n   * @see rpc.serverNotifications.roomAccessListRemoved\n   * @see rpc.serverNotifications.roomModeChanged\n   */\n  changeAccessListsUpdates (roomName, mode, cb) {\n    return this.state.getRoom(roomName)\n      .then(room => room.roomState.accessListsUpdatesSet(mode))\n      .return()\n      .asCallback(cb)\n  }\n\n  /**\n   * Enables or disables user list updates for the room.\n   *\n   * @param {string} roomName Room name.\n   * @param {boolean} mode Enable or disable.\n   * @param {callback} [cb] Optional callback.\n   *\n   * @return {Promise<undefined>} Promise that resolves without any data.\n   *\n   * @see rpc.serverNotifications.roomUserJoined\n   * @see rpc.serverNotifications.roomUserLeft\n   */\n  changeUserlistUpdates (roomName, mode, cb) {\n    return this.state.getRoom(roomName)\n      .then(room => room.roomState.userlistUpdatesSet(mode))\n      .return()\n      .asCallback(cb)\n  }\n\n}\n\nmodule.exports = ServiceAPI\n"]}
\No newline at end of file