ParseLiveQueryServer.js 131 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845
  1. "use strict";
  2. Object.defineProperty(exports, "__esModule", {
  3. value: true
  4. });
  5. exports.ParseLiveQueryServer = void 0;
  6. var _tv = _interopRequireDefault(require("tv4"));
  7. var _node = _interopRequireDefault(require("parse/node"));
  8. var _Subscription = require("./Subscription");
  9. var _Client = require("./Client");
  10. var _ParseWebSocketServer = require("./ParseWebSocketServer");
  11. var _logger = _interopRequireDefault(require("../logger"));
  12. var _RequestSchema = _interopRequireDefault(require("./RequestSchema"));
  13. var _QueryTools = require("./QueryTools");
  14. var _ParsePubSub = require("./ParsePubSub");
  15. var _SchemaController = _interopRequireDefault(require("../Controllers/SchemaController"));
  16. var _lodash = _interopRequireDefault(require("lodash"));
  17. var _uuid = require("uuid");
  18. var _triggers = require("../triggers");
  19. var _Auth = require("../Auth");
  20. var _Controllers = require("../Controllers");
  21. var _lruCache = require("lru-cache");
  22. var _UsersRouter = _interopRequireDefault(require("../Routers/UsersRouter"));
  23. var _DatabaseController = _interopRequireDefault(require("../Controllers/DatabaseController"));
  24. var _util = require("util");
  25. var _deepcopy = _interopRequireDefault(require("deepcopy"));
  26. function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
  27. class ParseLiveQueryServer {
  28. // className -> (queryHash -> subscription)
  29. // The subscriber we use to get object update from publisher
  30. constructor(server, config = {}, parseServerConfig = {}) {
  31. this.server = server;
  32. this.clients = new Map();
  33. this.subscriptions = new Map();
  34. this.config = config;
  35. config.appId = config.appId || _node.default.applicationId;
  36. config.masterKey = config.masterKey || _node.default.masterKey;
  37. // Store keys, convert obj to map
  38. const keyPairs = config.keyPairs || {};
  39. this.keyPairs = new Map();
  40. for (const key of Object.keys(keyPairs)) {
  41. this.keyPairs.set(key, keyPairs[key]);
  42. }
  43. _logger.default.verbose('Support key pairs', this.keyPairs);
  44. // Initialize Parse
  45. _node.default.Object.disableSingleInstance();
  46. const serverURL = config.serverURL || _node.default.serverURL;
  47. _node.default.serverURL = serverURL;
  48. _node.default.initialize(config.appId, _node.default.javaScriptKey, config.masterKey);
  49. // The cache controller is a proper cache controller
  50. // with access to User and Roles
  51. this.cacheController = (0, _Controllers.getCacheController)(parseServerConfig);
  52. config.cacheTimeout = config.cacheTimeout || 5 * 1000; // 5s
  53. // This auth cache stores the promises for each auth resolution.
  54. // The main benefit is to be able to reuse the same user / session token resolution.
  55. this.authCache = new _lruCache.LRUCache({
  56. max: 500,
  57. // 500 concurrent
  58. ttl: config.cacheTimeout
  59. });
  60. // Initialize websocket server
  61. this.parseWebSocketServer = new _ParseWebSocketServer.ParseWebSocketServer(server, parseWebsocket => this._onConnect(parseWebsocket), config);
  62. this.subscriber = _ParsePubSub.ParsePubSub.createSubscriber(config);
  63. if (!this.subscriber.connect) {
  64. this.connect();
  65. }
  66. }
  67. async connect() {
  68. if (this.subscriber.isOpen) {
  69. return;
  70. }
  71. if (typeof this.subscriber.connect === 'function') {
  72. await Promise.resolve(this.subscriber.connect());
  73. } else {
  74. this.subscriber.isOpen = true;
  75. }
  76. this._createSubscribers();
  77. }
  78. async shutdown() {
  79. if (this.subscriber.isOpen) {
  80. var _this$subscriber$clos, _this$subscriber;
  81. await Promise.all([...[...this.clients.values()].map(client => client.parseWebSocket.ws.close()), this.parseWebSocketServer.close(), ...Array.from(this.subscriber.subscriptions.keys()).map(key => this.subscriber.unsubscribe(key)), (_this$subscriber$clos = (_this$subscriber = this.subscriber).close) === null || _this$subscriber$clos === void 0 ? void 0 : _this$subscriber$clos.call(_this$subscriber)]);
  82. }
  83. this.subscriber.isOpen = false;
  84. }
  85. _createSubscribers() {
  86. const messageRecieved = (channel, messageStr) => {
  87. _logger.default.verbose('Subscribe message %j', messageStr);
  88. let message;
  89. try {
  90. message = JSON.parse(messageStr);
  91. } catch (e) {
  92. _logger.default.error('unable to parse message', messageStr, e);
  93. return;
  94. }
  95. if (channel === _node.default.applicationId + 'clearCache') {
  96. this._clearCachedRoles(message.userId);
  97. return;
  98. }
  99. this._inflateParseObject(message);
  100. if (channel === _node.default.applicationId + 'afterSave') {
  101. this._onAfterSave(message);
  102. } else if (channel === _node.default.applicationId + 'afterDelete') {
  103. this._onAfterDelete(message);
  104. } else {
  105. _logger.default.error('Get message %s from unknown channel %j', message, channel);
  106. }
  107. };
  108. this.subscriber.on('message', (channel, messageStr) => messageRecieved(channel, messageStr));
  109. for (const field of ['afterSave', 'afterDelete', 'clearCache']) {
  110. const channel = `${_node.default.applicationId}${field}`;
  111. this.subscriber.subscribe(channel, messageStr => messageRecieved(channel, messageStr));
  112. }
  113. }
  114. // Message is the JSON object from publisher. Message.currentParseObject is the ParseObject JSON after changes.
  115. // Message.originalParseObject is the original ParseObject JSON.
  116. _inflateParseObject(message) {
  117. // Inflate merged object
  118. const currentParseObject = message.currentParseObject;
  119. _UsersRouter.default.removeHiddenProperties(currentParseObject);
  120. let className = currentParseObject.className;
  121. let parseObject = new _node.default.Object(className);
  122. parseObject._finishFetch(currentParseObject);
  123. message.currentParseObject = parseObject;
  124. // Inflate original object
  125. const originalParseObject = message.originalParseObject;
  126. if (originalParseObject) {
  127. _UsersRouter.default.removeHiddenProperties(originalParseObject);
  128. className = originalParseObject.className;
  129. parseObject = new _node.default.Object(className);
  130. parseObject._finishFetch(originalParseObject);
  131. message.originalParseObject = parseObject;
  132. }
  133. }
  134. // Message is the JSON object from publisher after inflated. Message.currentParseObject is the ParseObject after changes.
  135. // Message.originalParseObject is the original ParseObject.
  136. async _onAfterDelete(message) {
  137. _logger.default.verbose(_node.default.applicationId + 'afterDelete is triggered');
  138. let deletedParseObject = message.currentParseObject.toJSON();
  139. const classLevelPermissions = message.classLevelPermissions;
  140. const className = deletedParseObject.className;
  141. _logger.default.verbose('ClassName: %j | ObjectId: %s', className, deletedParseObject.id);
  142. _logger.default.verbose('Current client number : %d', this.clients.size);
  143. const classSubscriptions = this.subscriptions.get(className);
  144. if (typeof classSubscriptions === 'undefined') {
  145. _logger.default.debug('Can not find subscriptions under this class ' + className);
  146. return;
  147. }
  148. for (const subscription of classSubscriptions.values()) {
  149. const isSubscriptionMatched = this._matchesSubscription(deletedParseObject, subscription);
  150. if (!isSubscriptionMatched) {
  151. continue;
  152. }
  153. for (const [clientId, requestIds] of _lodash.default.entries(subscription.clientRequestIds)) {
  154. const client = this.clients.get(clientId);
  155. if (typeof client === 'undefined') {
  156. continue;
  157. }
  158. requestIds.forEach(async requestId => {
  159. const acl = message.currentParseObject.getACL();
  160. // Check CLP
  161. const op = this._getCLPOperation(subscription.query);
  162. let res = {};
  163. try {
  164. await this._matchesCLP(classLevelPermissions, message.currentParseObject, client, requestId, op);
  165. const isMatched = await this._matchesACL(acl, client, requestId);
  166. if (!isMatched) {
  167. return null;
  168. }
  169. res = {
  170. event: 'delete',
  171. sessionToken: client.sessionToken,
  172. object: deletedParseObject,
  173. clients: this.clients.size,
  174. subscriptions: this.subscriptions.size,
  175. useMasterKey: client.hasMasterKey,
  176. installationId: client.installationId,
  177. sendEvent: true
  178. };
  179. const trigger = (0, _triggers.getTrigger)(className, 'afterEvent', _node.default.applicationId);
  180. if (trigger) {
  181. const auth = await this.getAuthFromClient(client, requestId);
  182. if (auth && auth.user) {
  183. res.user = auth.user;
  184. }
  185. if (res.object) {
  186. res.object = _node.default.Object.fromJSON(res.object);
  187. }
  188. await (0, _triggers.runTrigger)(trigger, `afterEvent.${className}`, res, auth);
  189. }
  190. if (!res.sendEvent) {
  191. return;
  192. }
  193. if (res.object && typeof res.object.toJSON === 'function') {
  194. deletedParseObject = (0, _triggers.toJSONwithObjects)(res.object, res.object.className || className);
  195. }
  196. await this._filterSensitiveData(classLevelPermissions, res, client, requestId, op, subscription.query);
  197. client.pushDelete(requestId, deletedParseObject);
  198. } catch (e) {
  199. const error = (0, _triggers.resolveError)(e);
  200. _Client.Client.pushError(client.parseWebSocket, error.code, error.message, false, requestId);
  201. _logger.default.error(`Failed running afterLiveQueryEvent on class ${className} for event ${res.event} with session ${res.sessionToken} with:\n Error: ` + JSON.stringify(error));
  202. }
  203. });
  204. }
  205. }
  206. }
  207. // Message is the JSON object from publisher after inflated. Message.currentParseObject is the ParseObject after changes.
  208. // Message.originalParseObject is the original ParseObject.
  209. async _onAfterSave(message) {
  210. _logger.default.verbose(_node.default.applicationId + 'afterSave is triggered');
  211. let originalParseObject = null;
  212. if (message.originalParseObject) {
  213. originalParseObject = message.originalParseObject.toJSON();
  214. }
  215. const classLevelPermissions = message.classLevelPermissions;
  216. let currentParseObject = message.currentParseObject.toJSON();
  217. const className = currentParseObject.className;
  218. _logger.default.verbose('ClassName: %s | ObjectId: %s', className, currentParseObject.id);
  219. _logger.default.verbose('Current client number : %d', this.clients.size);
  220. const classSubscriptions = this.subscriptions.get(className);
  221. if (typeof classSubscriptions === 'undefined') {
  222. _logger.default.debug('Can not find subscriptions under this class ' + className);
  223. return;
  224. }
  225. for (const subscription of classSubscriptions.values()) {
  226. const isOriginalSubscriptionMatched = this._matchesSubscription(originalParseObject, subscription);
  227. const isCurrentSubscriptionMatched = this._matchesSubscription(currentParseObject, subscription);
  228. for (const [clientId, requestIds] of _lodash.default.entries(subscription.clientRequestIds)) {
  229. const client = this.clients.get(clientId);
  230. if (typeof client === 'undefined') {
  231. continue;
  232. }
  233. requestIds.forEach(async requestId => {
  234. // Set orignal ParseObject ACL checking promise, if the object does not match
  235. // subscription, we do not need to check ACL
  236. let originalACLCheckingPromise;
  237. if (!isOriginalSubscriptionMatched) {
  238. originalACLCheckingPromise = Promise.resolve(false);
  239. } else {
  240. let originalACL;
  241. if (message.originalParseObject) {
  242. originalACL = message.originalParseObject.getACL();
  243. }
  244. originalACLCheckingPromise = this._matchesACL(originalACL, client, requestId);
  245. }
  246. // Set current ParseObject ACL checking promise, if the object does not match
  247. // subscription, we do not need to check ACL
  248. let currentACLCheckingPromise;
  249. let res = {};
  250. if (!isCurrentSubscriptionMatched) {
  251. currentACLCheckingPromise = Promise.resolve(false);
  252. } else {
  253. const currentACL = message.currentParseObject.getACL();
  254. currentACLCheckingPromise = this._matchesACL(currentACL, client, requestId);
  255. }
  256. try {
  257. const op = this._getCLPOperation(subscription.query);
  258. await this._matchesCLP(classLevelPermissions, message.currentParseObject, client, requestId, op);
  259. const [isOriginalMatched, isCurrentMatched] = await Promise.all([originalACLCheckingPromise, currentACLCheckingPromise]);
  260. _logger.default.verbose('Original %j | Current %j | Match: %s, %s, %s, %s | Query: %s', originalParseObject, currentParseObject, isOriginalSubscriptionMatched, isCurrentSubscriptionMatched, isOriginalMatched, isCurrentMatched, subscription.hash);
  261. // Decide event type
  262. let type;
  263. if (isOriginalMatched && isCurrentMatched) {
  264. type = 'update';
  265. } else if (isOriginalMatched && !isCurrentMatched) {
  266. type = 'leave';
  267. } else if (!isOriginalMatched && isCurrentMatched) {
  268. if (originalParseObject) {
  269. type = 'enter';
  270. } else {
  271. type = 'create';
  272. }
  273. } else {
  274. return null;
  275. }
  276. const watchFieldsChanged = this._checkWatchFields(client, requestId, message);
  277. if (!watchFieldsChanged && (type === 'update' || type === 'create')) {
  278. return;
  279. }
  280. res = {
  281. event: type,
  282. sessionToken: client.sessionToken,
  283. object: currentParseObject,
  284. original: originalParseObject,
  285. clients: this.clients.size,
  286. subscriptions: this.subscriptions.size,
  287. useMasterKey: client.hasMasterKey,
  288. installationId: client.installationId,
  289. sendEvent: true
  290. };
  291. const trigger = (0, _triggers.getTrigger)(className, 'afterEvent', _node.default.applicationId);
  292. if (trigger) {
  293. if (res.object) {
  294. res.object = _node.default.Object.fromJSON(res.object);
  295. }
  296. if (res.original) {
  297. res.original = _node.default.Object.fromJSON(res.original);
  298. }
  299. const auth = await this.getAuthFromClient(client, requestId);
  300. if (auth && auth.user) {
  301. res.user = auth.user;
  302. }
  303. await (0, _triggers.runTrigger)(trigger, `afterEvent.${className}`, res, auth);
  304. }
  305. if (!res.sendEvent) {
  306. return;
  307. }
  308. if (res.object && typeof res.object.toJSON === 'function') {
  309. currentParseObject = (0, _triggers.toJSONwithObjects)(res.object, res.object.className || className);
  310. }
  311. if (res.original && typeof res.original.toJSON === 'function') {
  312. originalParseObject = (0, _triggers.toJSONwithObjects)(res.original, res.original.className || className);
  313. }
  314. await this._filterSensitiveData(classLevelPermissions, res, client, requestId, op, subscription.query);
  315. const functionName = 'push' + res.event.charAt(0).toUpperCase() + res.event.slice(1);
  316. if (client[functionName]) {
  317. client[functionName](requestId, currentParseObject, originalParseObject);
  318. }
  319. } catch (e) {
  320. const error = (0, _triggers.resolveError)(e);
  321. _Client.Client.pushError(client.parseWebSocket, error.code, error.message, false, requestId);
  322. _logger.default.error(`Failed running afterLiveQueryEvent on class ${className} for event ${res.event} with session ${res.sessionToken} with:\n Error: ` + JSON.stringify(error));
  323. }
  324. });
  325. }
  326. }
  327. }
  328. _onConnect(parseWebsocket) {
  329. parseWebsocket.on('message', request => {
  330. if (typeof request === 'string') {
  331. try {
  332. request = JSON.parse(request);
  333. } catch (e) {
  334. _logger.default.error('unable to parse request', request, e);
  335. return;
  336. }
  337. }
  338. _logger.default.verbose('Request: %j', request);
  339. // Check whether this request is a valid request, return error directly if not
  340. if (!_tv.default.validate(request, _RequestSchema.default['general']) || !_tv.default.validate(request, _RequestSchema.default[request.op])) {
  341. _Client.Client.pushError(parseWebsocket, 1, _tv.default.error.message);
  342. _logger.default.error('Connect message error %s', _tv.default.error.message);
  343. return;
  344. }
  345. switch (request.op) {
  346. case 'connect':
  347. this._handleConnect(parseWebsocket, request);
  348. break;
  349. case 'subscribe':
  350. this._handleSubscribe(parseWebsocket, request);
  351. break;
  352. case 'update':
  353. this._handleUpdateSubscription(parseWebsocket, request);
  354. break;
  355. case 'unsubscribe':
  356. this._handleUnsubscribe(parseWebsocket, request);
  357. break;
  358. default:
  359. _Client.Client.pushError(parseWebsocket, 3, 'Get unknown operation');
  360. _logger.default.error('Get unknown operation', request.op);
  361. }
  362. });
  363. parseWebsocket.on('disconnect', () => {
  364. _logger.default.info(`Client disconnect: ${parseWebsocket.clientId}`);
  365. const clientId = parseWebsocket.clientId;
  366. if (!this.clients.has(clientId)) {
  367. (0, _triggers.runLiveQueryEventHandlers)({
  368. event: 'ws_disconnect_error',
  369. clients: this.clients.size,
  370. subscriptions: this.subscriptions.size,
  371. error: `Unable to find client ${clientId}`
  372. });
  373. _logger.default.error(`Can not find client ${clientId} on disconnect`);
  374. return;
  375. }
  376. // Delete client
  377. const client = this.clients.get(clientId);
  378. this.clients.delete(clientId);
  379. // Delete client from subscriptions
  380. for (const [requestId, subscriptionInfo] of _lodash.default.entries(client.subscriptionInfos)) {
  381. const subscription = subscriptionInfo.subscription;
  382. subscription.deleteClientSubscription(clientId, requestId);
  383. // If there is no client which is subscribing this subscription, remove it from subscriptions
  384. const classSubscriptions = this.subscriptions.get(subscription.className);
  385. if (!subscription.hasSubscribingClient()) {
  386. classSubscriptions.delete(subscription.hash);
  387. }
  388. // If there is no subscriptions under this class, remove it from subscriptions
  389. if (classSubscriptions.size === 0) {
  390. this.subscriptions.delete(subscription.className);
  391. }
  392. }
  393. _logger.default.verbose('Current clients %d', this.clients.size);
  394. _logger.default.verbose('Current subscriptions %d', this.subscriptions.size);
  395. (0, _triggers.runLiveQueryEventHandlers)({
  396. event: 'ws_disconnect',
  397. clients: this.clients.size,
  398. subscriptions: this.subscriptions.size,
  399. useMasterKey: client.hasMasterKey,
  400. installationId: client.installationId,
  401. sessionToken: client.sessionToken
  402. });
  403. });
  404. (0, _triggers.runLiveQueryEventHandlers)({
  405. event: 'ws_connect',
  406. clients: this.clients.size,
  407. subscriptions: this.subscriptions.size
  408. });
  409. }
  410. _matchesSubscription(parseObject, subscription) {
  411. // Object is undefined or null, not match
  412. if (!parseObject) {
  413. return false;
  414. }
  415. return (0, _QueryTools.matchesQuery)((0, _deepcopy.default)(parseObject), subscription.query);
  416. }
  417. async _clearCachedRoles(userId) {
  418. try {
  419. const validTokens = await new _node.default.Query(_node.default.Session).equalTo('user', _node.default.User.createWithoutData(userId)).find({
  420. useMasterKey: true
  421. });
  422. await Promise.all(validTokens.map(async token => {
  423. var _auth1$auth, _auth2$auth;
  424. const sessionToken = token.get('sessionToken');
  425. const authPromise = this.authCache.get(sessionToken);
  426. if (!authPromise) {
  427. return;
  428. }
  429. const [auth1, auth2] = await Promise.all([authPromise, (0, _Auth.getAuthForSessionToken)({
  430. cacheController: this.cacheController,
  431. sessionToken
  432. })]);
  433. (_auth1$auth = auth1.auth) === null || _auth1$auth === void 0 || _auth1$auth.clearRoleCache(sessionToken);
  434. (_auth2$auth = auth2.auth) === null || _auth2$auth === void 0 || _auth2$auth.clearRoleCache(sessionToken);
  435. this.authCache.delete(sessionToken);
  436. }));
  437. } catch (e) {
  438. _logger.default.verbose(`Could not clear role cache. ${e}`);
  439. }
  440. }
  441. getAuthForSessionToken(sessionToken) {
  442. if (!sessionToken) {
  443. return Promise.resolve({});
  444. }
  445. const fromCache = this.authCache.get(sessionToken);
  446. if (fromCache) {
  447. return fromCache;
  448. }
  449. const authPromise = (0, _Auth.getAuthForSessionToken)({
  450. cacheController: this.cacheController,
  451. sessionToken: sessionToken
  452. }).then(auth => {
  453. return {
  454. auth,
  455. userId: auth && auth.user && auth.user.id
  456. };
  457. }).catch(error => {
  458. // There was an error with the session token
  459. const result = {};
  460. if (error && error.code === _node.default.Error.INVALID_SESSION_TOKEN) {
  461. result.error = error;
  462. this.authCache.set(sessionToken, Promise.resolve(result), this.config.cacheTimeout);
  463. } else {
  464. this.authCache.delete(sessionToken);
  465. }
  466. return result;
  467. });
  468. this.authCache.set(sessionToken, authPromise);
  469. return authPromise;
  470. }
  471. async _matchesCLP(classLevelPermissions, object, client, requestId, op) {
  472. // try to match on user first, less expensive than with roles
  473. const subscriptionInfo = client.getSubscriptionInfo(requestId);
  474. const aclGroup = ['*'];
  475. let userId;
  476. if (typeof subscriptionInfo !== 'undefined') {
  477. const {
  478. userId
  479. } = await this.getAuthForSessionToken(subscriptionInfo.sessionToken);
  480. if (userId) {
  481. aclGroup.push(userId);
  482. }
  483. }
  484. try {
  485. await _SchemaController.default.validatePermission(classLevelPermissions, object.className, aclGroup, op);
  486. return true;
  487. } catch (e) {
  488. _logger.default.verbose(`Failed matching CLP for ${object.id} ${userId} ${e}`);
  489. return false;
  490. }
  491. // TODO: handle roles permissions
  492. // Object.keys(classLevelPermissions).forEach((key) => {
  493. // const perm = classLevelPermissions[key];
  494. // Object.keys(perm).forEach((key) => {
  495. // if (key.indexOf('role'))
  496. // });
  497. // })
  498. // // it's rejected here, check the roles
  499. // var rolesQuery = new Parse.Query(Parse.Role);
  500. // rolesQuery.equalTo("users", user);
  501. // return rolesQuery.find({useMasterKey:true});
  502. }
  503. async _filterSensitiveData(classLevelPermissions, res, client, requestId, op, query) {
  504. const subscriptionInfo = client.getSubscriptionInfo(requestId);
  505. const aclGroup = ['*'];
  506. let clientAuth;
  507. if (typeof subscriptionInfo !== 'undefined') {
  508. const {
  509. userId,
  510. auth
  511. } = await this.getAuthForSessionToken(subscriptionInfo.sessionToken);
  512. if (userId) {
  513. aclGroup.push(userId);
  514. }
  515. clientAuth = auth;
  516. }
  517. const filter = obj => {
  518. if (!obj) {
  519. return;
  520. }
  521. let protectedFields = (classLevelPermissions === null || classLevelPermissions === void 0 ? void 0 : classLevelPermissions.protectedFields) || [];
  522. if (!client.hasMasterKey && !Array.isArray(protectedFields)) {
  523. protectedFields = (0, _Controllers.getDatabaseController)(this.config).addProtectedFields(classLevelPermissions, res.object.className, query, aclGroup, clientAuth);
  524. }
  525. return _DatabaseController.default.filterSensitiveData(client.hasMasterKey, false, aclGroup, clientAuth, op, classLevelPermissions, res.object.className, protectedFields, obj, query);
  526. };
  527. res.object = filter(res.object);
  528. res.original = filter(res.original);
  529. }
  530. _getCLPOperation(query) {
  531. return typeof query === 'object' && Object.keys(query).length == 1 && typeof query.objectId === 'string' ? 'get' : 'find';
  532. }
  533. async _verifyACL(acl, token) {
  534. if (!token) {
  535. return false;
  536. }
  537. const {
  538. auth,
  539. userId
  540. } = await this.getAuthForSessionToken(token);
  541. // Getting the session token failed
  542. // This means that no additional auth is available
  543. // At this point, just bail out as no additional visibility can be inferred.
  544. if (!auth || !userId) {
  545. return false;
  546. }
  547. const isSubscriptionSessionTokenMatched = acl.getReadAccess(userId);
  548. if (isSubscriptionSessionTokenMatched) {
  549. return true;
  550. }
  551. // Check if the user has any roles that match the ACL
  552. return Promise.resolve().then(async () => {
  553. // Resolve false right away if the acl doesn't have any roles
  554. const acl_has_roles = Object.keys(acl.permissionsById).some(key => key.startsWith('role:'));
  555. if (!acl_has_roles) {
  556. return false;
  557. }
  558. const roleNames = await auth.getUserRoles();
  559. // Finally, see if any of the user's roles allow them read access
  560. for (const role of roleNames) {
  561. // We use getReadAccess as `role` is in the form `role:roleName`
  562. if (acl.getReadAccess(role)) {
  563. return true;
  564. }
  565. }
  566. return false;
  567. }).catch(() => {
  568. return false;
  569. });
  570. }
  571. async getAuthFromClient(client, requestId, sessionToken) {
  572. const getSessionFromClient = () => {
  573. const subscriptionInfo = client.getSubscriptionInfo(requestId);
  574. if (typeof subscriptionInfo === 'undefined') {
  575. return client.sessionToken;
  576. }
  577. return subscriptionInfo.sessionToken || client.sessionToken;
  578. };
  579. if (!sessionToken) {
  580. sessionToken = getSessionFromClient();
  581. }
  582. if (!sessionToken) {
  583. return;
  584. }
  585. const {
  586. auth
  587. } = await this.getAuthForSessionToken(sessionToken);
  588. return auth;
  589. }
  590. _checkWatchFields(client, requestId, message) {
  591. const subscriptionInfo = client.getSubscriptionInfo(requestId);
  592. const watch = subscriptionInfo === null || subscriptionInfo === void 0 ? void 0 : subscriptionInfo.watch;
  593. if (!watch) {
  594. return true;
  595. }
  596. const object = message.currentParseObject;
  597. const original = message.originalParseObject;
  598. return watch.some(field => !(0, _util.isDeepStrictEqual)(object.get(field), original === null || original === void 0 ? void 0 : original.get(field)));
  599. }
  600. async _matchesACL(acl, client, requestId) {
  601. // Return true directly if ACL isn't present, ACL is public read, or client has master key
  602. if (!acl || acl.getPublicReadAccess() || client.hasMasterKey) {
  603. return true;
  604. }
  605. // Check subscription sessionToken matches ACL first
  606. const subscriptionInfo = client.getSubscriptionInfo(requestId);
  607. if (typeof subscriptionInfo === 'undefined') {
  608. return false;
  609. }
  610. const subscriptionToken = subscriptionInfo.sessionToken;
  611. const clientSessionToken = client.sessionToken;
  612. if (await this._verifyACL(acl, subscriptionToken)) {
  613. return true;
  614. }
  615. if (await this._verifyACL(acl, clientSessionToken)) {
  616. return true;
  617. }
  618. return false;
  619. }
  620. async _handleConnect(parseWebsocket, request) {
  621. if (!this._validateKeys(request, this.keyPairs)) {
  622. _Client.Client.pushError(parseWebsocket, 4, 'Key in request is not valid');
  623. _logger.default.error('Key in request is not valid');
  624. return;
  625. }
  626. const hasMasterKey = this._hasMasterKey(request, this.keyPairs);
  627. const clientId = (0, _uuid.v4)();
  628. const client = new _Client.Client(clientId, parseWebsocket, hasMasterKey, request.sessionToken, request.installationId);
  629. try {
  630. const req = {
  631. client,
  632. event: 'connect',
  633. clients: this.clients.size,
  634. subscriptions: this.subscriptions.size,
  635. sessionToken: request.sessionToken,
  636. useMasterKey: client.hasMasterKey,
  637. installationId: request.installationId
  638. };
  639. const trigger = (0, _triggers.getTrigger)('@Connect', 'beforeConnect', _node.default.applicationId);
  640. if (trigger) {
  641. const auth = await this.getAuthFromClient(client, request.requestId, req.sessionToken);
  642. if (auth && auth.user) {
  643. req.user = auth.user;
  644. }
  645. await (0, _triggers.runTrigger)(trigger, `beforeConnect.@Connect`, req, auth);
  646. }
  647. parseWebsocket.clientId = clientId;
  648. this.clients.set(parseWebsocket.clientId, client);
  649. _logger.default.info(`Create new client: ${parseWebsocket.clientId}`);
  650. client.pushConnect();
  651. (0, _triggers.runLiveQueryEventHandlers)(req);
  652. } catch (e) {
  653. const error = (0, _triggers.resolveError)(e);
  654. _Client.Client.pushError(parseWebsocket, error.code, error.message, false);
  655. _logger.default.error(`Failed running beforeConnect for session ${request.sessionToken} with:\n Error: ` + JSON.stringify(error));
  656. }
  657. }
  658. _hasMasterKey(request, validKeyPairs) {
  659. if (!validKeyPairs || validKeyPairs.size == 0 || !validKeyPairs.has('masterKey')) {
  660. return false;
  661. }
  662. if (!request || !Object.prototype.hasOwnProperty.call(request, 'masterKey')) {
  663. return false;
  664. }
  665. return request.masterKey === validKeyPairs.get('masterKey');
  666. }
  667. _validateKeys(request, validKeyPairs) {
  668. if (!validKeyPairs || validKeyPairs.size == 0) {
  669. return true;
  670. }
  671. let isValid = false;
  672. for (const [key, secret] of validKeyPairs) {
  673. if (!request[key] || request[key] !== secret) {
  674. continue;
  675. }
  676. isValid = true;
  677. break;
  678. }
  679. return isValid;
  680. }
  681. async _handleSubscribe(parseWebsocket, request) {
  682. // If we can not find this client, return error to client
  683. if (!Object.prototype.hasOwnProperty.call(parseWebsocket, 'clientId')) {
  684. _Client.Client.pushError(parseWebsocket, 2, 'Can not find this client, make sure you connect to server before subscribing');
  685. _logger.default.error('Can not find this client, make sure you connect to server before subscribing');
  686. return;
  687. }
  688. const client = this.clients.get(parseWebsocket.clientId);
  689. const className = request.query.className;
  690. let authCalled = false;
  691. try {
  692. const trigger = (0, _triggers.getTrigger)(className, 'beforeSubscribe', _node.default.applicationId);
  693. if (trigger) {
  694. const auth = await this.getAuthFromClient(client, request.requestId, request.sessionToken);
  695. authCalled = true;
  696. if (auth && auth.user) {
  697. request.user = auth.user;
  698. }
  699. const parseQuery = new _node.default.Query(className);
  700. parseQuery.withJSON(request.query);
  701. request.query = parseQuery;
  702. await (0, _triggers.runTrigger)(trigger, `beforeSubscribe.${className}`, request, auth);
  703. const query = request.query.toJSON();
  704. request.query = query;
  705. }
  706. if (className === '_Session') {
  707. if (!authCalled) {
  708. const auth = await this.getAuthFromClient(client, request.requestId, request.sessionToken);
  709. if (auth && auth.user) {
  710. request.user = auth.user;
  711. }
  712. }
  713. if (request.user) {
  714. request.query.where.user = request.user.toPointer();
  715. } else if (!request.master) {
  716. _Client.Client.pushError(parseWebsocket, _node.default.Error.INVALID_SESSION_TOKEN, 'Invalid session token', false, request.requestId);
  717. return;
  718. }
  719. }
  720. // Get subscription from subscriptions, create one if necessary
  721. const subscriptionHash = (0, _QueryTools.queryHash)(request.query);
  722. // Add className to subscriptions if necessary
  723. if (!this.subscriptions.has(className)) {
  724. this.subscriptions.set(className, new Map());
  725. }
  726. const classSubscriptions = this.subscriptions.get(className);
  727. let subscription;
  728. if (classSubscriptions.has(subscriptionHash)) {
  729. subscription = classSubscriptions.get(subscriptionHash);
  730. } else {
  731. subscription = new _Subscription.Subscription(className, request.query.where, subscriptionHash);
  732. classSubscriptions.set(subscriptionHash, subscription);
  733. }
  734. // Add subscriptionInfo to client
  735. const subscriptionInfo = {
  736. subscription: subscription
  737. };
  738. // Add selected fields, sessionToken and installationId for this subscription if necessary
  739. if (request.query.keys) {
  740. subscriptionInfo.keys = Array.isArray(request.query.keys) ? request.query.keys : request.query.keys.split(',');
  741. }
  742. if (request.query.watch) {
  743. subscriptionInfo.watch = request.query.watch;
  744. }
  745. if (request.sessionToken) {
  746. subscriptionInfo.sessionToken = request.sessionToken;
  747. }
  748. client.addSubscriptionInfo(request.requestId, subscriptionInfo);
  749. // Add clientId to subscription
  750. subscription.addClientSubscription(parseWebsocket.clientId, request.requestId);
  751. client.pushSubscribe(request.requestId);
  752. _logger.default.verbose(`Create client ${parseWebsocket.clientId} new subscription: ${request.requestId}`);
  753. _logger.default.verbose('Current client number: %d', this.clients.size);
  754. (0, _triggers.runLiveQueryEventHandlers)({
  755. client,
  756. event: 'subscribe',
  757. clients: this.clients.size,
  758. subscriptions: this.subscriptions.size,
  759. sessionToken: request.sessionToken,
  760. useMasterKey: client.hasMasterKey,
  761. installationId: client.installationId
  762. });
  763. } catch (e) {
  764. const error = (0, _triggers.resolveError)(e);
  765. _Client.Client.pushError(parseWebsocket, error.code, error.message, false, request.requestId);
  766. _logger.default.error(`Failed running beforeSubscribe on ${className} for session ${request.sessionToken} with:\n Error: ` + JSON.stringify(error));
  767. }
  768. }
  769. _handleUpdateSubscription(parseWebsocket, request) {
  770. this._handleUnsubscribe(parseWebsocket, request, false);
  771. this._handleSubscribe(parseWebsocket, request);
  772. }
  773. _handleUnsubscribe(parseWebsocket, request, notifyClient = true) {
  774. // If we can not find this client, return error to client
  775. if (!Object.prototype.hasOwnProperty.call(parseWebsocket, 'clientId')) {
  776. _Client.Client.pushError(parseWebsocket, 2, 'Can not find this client, make sure you connect to server before unsubscribing');
  777. _logger.default.error('Can not find this client, make sure you connect to server before unsubscribing');
  778. return;
  779. }
  780. const requestId = request.requestId;
  781. const client = this.clients.get(parseWebsocket.clientId);
  782. if (typeof client === 'undefined') {
  783. _Client.Client.pushError(parseWebsocket, 2, 'Cannot find client with clientId ' + parseWebsocket.clientId + '. Make sure you connect to live query server before unsubscribing.');
  784. _logger.default.error('Can not find this client ' + parseWebsocket.clientId);
  785. return;
  786. }
  787. const subscriptionInfo = client.getSubscriptionInfo(requestId);
  788. if (typeof subscriptionInfo === 'undefined') {
  789. _Client.Client.pushError(parseWebsocket, 2, 'Cannot find subscription with clientId ' + parseWebsocket.clientId + ' subscriptionId ' + requestId + '. Make sure you subscribe to live query server before unsubscribing.');
  790. _logger.default.error('Can not find subscription with clientId ' + parseWebsocket.clientId + ' subscriptionId ' + requestId);
  791. return;
  792. }
  793. // Remove subscription from client
  794. client.deleteSubscriptionInfo(requestId);
  795. // Remove client from subscription
  796. const subscription = subscriptionInfo.subscription;
  797. const className = subscription.className;
  798. subscription.deleteClientSubscription(parseWebsocket.clientId, requestId);
  799. // If there is no client which is subscribing this subscription, remove it from subscriptions
  800. const classSubscriptions = this.subscriptions.get(className);
  801. if (!subscription.hasSubscribingClient()) {
  802. classSubscriptions.delete(subscription.hash);
  803. }
  804. // If there is no subscriptions under this class, remove it from subscriptions
  805. if (classSubscriptions.size === 0) {
  806. this.subscriptions.delete(className);
  807. }
  808. (0, _triggers.runLiveQueryEventHandlers)({
  809. client,
  810. event: 'unsubscribe',
  811. clients: this.clients.size,
  812. subscriptions: this.subscriptions.size,
  813. sessionToken: subscriptionInfo.sessionToken,
  814. useMasterKey: client.hasMasterKey,
  815. installationId: client.installationId
  816. });
  817. if (!notifyClient) {
  818. return;
  819. }
  820. client.pushUnsubscribe(request.requestId);
  821. _logger.default.verbose(`Delete client: ${parseWebsocket.clientId} | subscription: ${request.requestId}`);
  822. }
  823. }
  824. exports.ParseLiveQueryServer = ParseLiveQueryServer;
  825. //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_tv","_interopRequireDefault","require","_node","_Subscription","_Client","_ParseWebSocketServer","_logger","_RequestSchema","_QueryTools","_ParsePubSub","_SchemaController","_lodash","_uuid","_triggers","_Auth","_Controllers","_lruCache","_UsersRouter","_DatabaseController","_util","_deepcopy","e","__esModule","default","ParseLiveQueryServer","constructor","server","config","parseServerConfig","clients","Map","subscriptions","appId","Parse","applicationId","masterKey","keyPairs","key","Object","keys","set","logger","verbose","disableSingleInstance","serverURL","initialize","javaScriptKey","cacheController","getCacheController","cacheTimeout","authCache","LRU","max","ttl","parseWebSocketServer","ParseWebSocketServer","parseWebsocket","_onConnect","subscriber","ParsePubSub","createSubscriber","connect","isOpen","Promise","resolve","_createSubscribers","shutdown","_this$subscriber$clos","_this$subscriber","all","values","map","client","parseWebSocket","ws","close","Array","from","unsubscribe","call","messageRecieved","channel","messageStr","message","JSON","parse","error","_clearCachedRoles","userId","_inflateParseObject","_onAfterSave","_onAfterDelete","on","field","subscribe","currentParseObject","UserRouter","removeHiddenProperties","className","parseObject","_finishFetch","originalParseObject","deletedParseObject","toJSON","classLevelPermissions","id","size","classSubscriptions","get","debug","subscription","isSubscriptionMatched","_matchesSubscription","clientId","requestIds","_","entries","clientRequestIds","forEach","requestId","acl","getACL","op","_getCLPOperation","query","res","_matchesCLP","isMatched","_matchesACL","event","sessionToken","object","useMasterKey","hasMasterKey","installationId","sendEvent","trigger","getTrigger","auth","getAuthFromClient","user","fromJSON","runTrigger","toJSONwithObjects","_filterSensitiveData","pushDelete","resolveError","Client","pushError","code","stringify","isOriginalSubscriptionMatched","isCurrentSubscriptionMatched","originalACLCheckingPromise","originalACL","currentACLCheckingPromise","currentACL","isOriginalMatched","isCurrentMatched","hash","type","watchFieldsChanged","_checkWatchFields","original","functionName","charAt","toUpperCase","slice","request","tv4","validate","RequestSchema","_handleConnect","_handleSubscribe","_handleUpdateSubscription","_handleUnsubscribe","info","has","runLiveQueryEventHandlers","delete","subscriptionInfo","subscriptionInfos","deleteClientSubscription","hasSubscribingClient","matchesQuery","deepcopy","validTokens","Query","Session","equalTo","User","createWithoutData","find","token","_auth1$auth","_auth2$auth","authPromise","auth1","auth2","getAuthForSessionToken","clearRoleCache","fromCache","then","catch","result","Error","INVALID_SESSION_TOKEN","getSubscriptionInfo","aclGroup","push","SchemaController","validatePermission","clientAuth","filter","obj","protectedFields","isArray","getDatabaseController","addProtectedFields","DatabaseController","filterSensitiveData","length","objectId","_verifyACL","isSubscriptionSessionTokenMatched","getReadAccess","acl_has_roles","permissionsById","some","startsWith","roleNames","getUserRoles","role","getSessionFromClient","watch","isDeepStrictEqual","getPublicReadAccess","subscriptionToken","clientSessionToken","_validateKeys","_hasMasterKey","uuidv4","req","pushConnect","validKeyPairs","prototype","hasOwnProperty","isValid","secret","authCalled","parseQuery","withJSON","where","toPointer","master","subscriptionHash","queryHash","Subscription","split","addSubscriptionInfo","addClientSubscription","pushSubscribe","notifyClient","deleteSubscriptionInfo","pushUnsubscribe","exports"],"sources":["../../src/LiveQuery/ParseLiveQueryServer.js"],"sourcesContent":["import tv4 from 'tv4';\nimport Parse from 'parse/node';\nimport { Subscription } from './Subscription';\nimport { Client } from './Client';\nimport { ParseWebSocketServer } from './ParseWebSocketServer';\nimport logger from '../logger';\nimport RequestSchema from './RequestSchema';\nimport { matchesQuery, queryHash } from './QueryTools';\nimport { ParsePubSub } from './ParsePubSub';\nimport SchemaController from '../Controllers/SchemaController';\nimport _ from 'lodash';\nimport { v4 as uuidv4 } from 'uuid';\nimport {\n  runLiveQueryEventHandlers,\n  getTrigger,\n  runTrigger,\n  resolveError,\n  toJSONwithObjects,\n} from '../triggers';\nimport { getAuthForSessionToken, Auth } from '../Auth';\nimport { getCacheController, getDatabaseController } from '../Controllers';\nimport { LRUCache as LRU } from 'lru-cache';\nimport UserRouter from '../Routers/UsersRouter';\nimport DatabaseController from '../Controllers/DatabaseController';\nimport { isDeepStrictEqual } from 'util';\nimport deepcopy from 'deepcopy';\n\nclass ParseLiveQueryServer {\n  clients: Map;\n  // className -> (queryHash -> subscription)\n  subscriptions: Object;\n  parseWebSocketServer: Object;\n  keyPairs: any;\n  // The subscriber we use to get object update from publisher\n  subscriber: Object;\n\n  constructor(server: any, config: any = {}, parseServerConfig: any = {}) {\n    this.server = server;\n    this.clients = new Map();\n    this.subscriptions = new Map();\n    this.config = config;\n\n    config.appId = config.appId || Parse.applicationId;\n    config.masterKey = config.masterKey || Parse.masterKey;\n\n    // Store keys, convert obj to map\n    const keyPairs = config.keyPairs || {};\n    this.keyPairs = new Map();\n    for (const key of Object.keys(keyPairs)) {\n      this.keyPairs.set(key, keyPairs[key]);\n    }\n    logger.verbose('Support key pairs', this.keyPairs);\n\n    // Initialize Parse\n    Parse.Object.disableSingleInstance();\n    const serverURL = config.serverURL || Parse.serverURL;\n    Parse.serverURL = serverURL;\n    Parse.initialize(config.appId, Parse.javaScriptKey, config.masterKey);\n\n    // The cache controller is a proper cache controller\n    // with access to User and Roles\n    this.cacheController = getCacheController(parseServerConfig);\n\n    config.cacheTimeout = config.cacheTimeout || 5 * 1000; // 5s\n\n    // This auth cache stores the promises for each auth resolution.\n    // The main benefit is to be able to reuse the same user / session token resolution.\n    this.authCache = new LRU({\n      max: 500, // 500 concurrent\n      ttl: config.cacheTimeout,\n    });\n    // Initialize websocket server\n    this.parseWebSocketServer = new ParseWebSocketServer(\n      server,\n      parseWebsocket => this._onConnect(parseWebsocket),\n      config\n    );\n    this.subscriber = ParsePubSub.createSubscriber(config);\n    if (!this.subscriber.connect) {\n      this.connect();\n    }\n  }\n\n  async connect() {\n    if (this.subscriber.isOpen) {\n      return;\n    }\n    if (typeof this.subscriber.connect === 'function') {\n      await Promise.resolve(this.subscriber.connect());\n    } else {\n      this.subscriber.isOpen = true;\n    }\n    this._createSubscribers();\n  }\n\n  async shutdown() {\n    if (this.subscriber.isOpen) {\n      await Promise.all([\n        ...[...this.clients.values()].map(client => client.parseWebSocket.ws.close()),\n        this.parseWebSocketServer.close(),\n        ...Array.from(this.subscriber.subscriptions.keys()).map(key =>\n          this.subscriber.unsubscribe(key)\n        ),\n        this.subscriber.close?.(),\n      ]);\n    }\n    this.subscriber.isOpen = false;\n  }\n\n  _createSubscribers() {\n    const messageRecieved = (channel, messageStr) => {\n      logger.verbose('Subscribe message %j', messageStr);\n      let message;\n      try {\n        message = JSON.parse(messageStr);\n      } catch (e) {\n        logger.error('unable to parse message', messageStr, e);\n        return;\n      }\n      if (channel === Parse.applicationId + 'clearCache') {\n        this._clearCachedRoles(message.userId);\n        return;\n      }\n      this._inflateParseObject(message);\n      if (channel === Parse.applicationId + 'afterSave') {\n        this._onAfterSave(message);\n      } else if (channel === Parse.applicationId + 'afterDelete') {\n        this._onAfterDelete(message);\n      } else {\n        logger.error('Get message %s from unknown channel %j', message, channel);\n      }\n    };\n    this.subscriber.on('message', (channel, messageStr) => messageRecieved(channel, messageStr));\n    for (const field of ['afterSave', 'afterDelete', 'clearCache']) {\n      const channel = `${Parse.applicationId}${field}`;\n      this.subscriber.subscribe(channel, messageStr => messageRecieved(channel, messageStr));\n    }\n  }\n\n  // Message is the JSON object from publisher. Message.currentParseObject is the ParseObject JSON after changes.\n  // Message.originalParseObject is the original ParseObject JSON.\n  _inflateParseObject(message: any): void {\n    // Inflate merged object\n    const currentParseObject = message.currentParseObject;\n    UserRouter.removeHiddenProperties(currentParseObject);\n    let className = currentParseObject.className;\n    let parseObject = new Parse.Object(className);\n    parseObject._finishFetch(currentParseObject);\n    message.currentParseObject = parseObject;\n    // Inflate original object\n    const originalParseObject = message.originalParseObject;\n    if (originalParseObject) {\n      UserRouter.removeHiddenProperties(originalParseObject);\n      className = originalParseObject.className;\n      parseObject = new Parse.Object(className);\n      parseObject._finishFetch(originalParseObject);\n      message.originalParseObject = parseObject;\n    }\n  }\n\n  // Message is the JSON object from publisher after inflated. Message.currentParseObject is the ParseObject after changes.\n  // Message.originalParseObject is the original ParseObject.\n  async _onAfterDelete(message: any): void {\n    logger.verbose(Parse.applicationId + 'afterDelete is triggered');\n\n    let deletedParseObject = message.currentParseObject.toJSON();\n    const classLevelPermissions = message.classLevelPermissions;\n    const className = deletedParseObject.className;\n    logger.verbose('ClassName: %j | ObjectId: %s', className, deletedParseObject.id);\n    logger.verbose('Current client number : %d', this.clients.size);\n\n    const classSubscriptions = this.subscriptions.get(className);\n    if (typeof classSubscriptions === 'undefined') {\n      logger.debug('Can not find subscriptions under this class ' + className);\n      return;\n    }\n\n    for (const subscription of classSubscriptions.values()) {\n      const isSubscriptionMatched = this._matchesSubscription(deletedParseObject, subscription);\n      if (!isSubscriptionMatched) {\n        continue;\n      }\n      for (const [clientId, requestIds] of _.entries(subscription.clientRequestIds)) {\n        const client = this.clients.get(clientId);\n        if (typeof client === 'undefined') {\n          continue;\n        }\n        requestIds.forEach(async requestId => {\n          const acl = message.currentParseObject.getACL();\n          // Check CLP\n          const op = this._getCLPOperation(subscription.query);\n          let res = {};\n          try {\n            await this._matchesCLP(\n              classLevelPermissions,\n              message.currentParseObject,\n              client,\n              requestId,\n              op\n            );\n            const isMatched = await this._matchesACL(acl, client, requestId);\n            if (!isMatched) {\n              return null;\n            }\n            res = {\n              event: 'delete',\n              sessionToken: client.sessionToken,\n              object: deletedParseObject,\n              clients: this.clients.size,\n              subscriptions: this.subscriptions.size,\n              useMasterKey: client.hasMasterKey,\n              installationId: client.installationId,\n              sendEvent: true,\n            };\n            const trigger = getTrigger(className, 'afterEvent', Parse.applicationId);\n            if (trigger) {\n              const auth = await this.getAuthFromClient(client, requestId);\n              if (auth && auth.user) {\n                res.user = auth.user;\n              }\n              if (res.object) {\n                res.object = Parse.Object.fromJSON(res.object);\n              }\n              await runTrigger(trigger, `afterEvent.${className}`, res, auth);\n            }\n            if (!res.sendEvent) {\n              return;\n            }\n            if (res.object && typeof res.object.toJSON === 'function') {\n              deletedParseObject = toJSONwithObjects(res.object, res.object.className || className);\n            }\n            await this._filterSensitiveData(\n              classLevelPermissions,\n              res,\n              client,\n              requestId,\n              op,\n              subscription.query\n            );\n            client.pushDelete(requestId, deletedParseObject);\n          } catch (e) {\n            const error = resolveError(e);\n            Client.pushError(client.parseWebSocket, error.code, error.message, false, requestId);\n            logger.error(\n              `Failed running afterLiveQueryEvent on class ${className} for event ${res.event} with session ${res.sessionToken} with:\\n Error: ` +\n                JSON.stringify(error)\n            );\n          }\n        });\n      }\n    }\n  }\n\n  // Message is the JSON object from publisher after inflated. Message.currentParseObject is the ParseObject after changes.\n  // Message.originalParseObject is the original ParseObject.\n  async _onAfterSave(message: any): void {\n    logger.verbose(Parse.applicationId + 'afterSave is triggered');\n\n    let originalParseObject = null;\n    if (message.originalParseObject) {\n      originalParseObject = message.originalParseObject.toJSON();\n    }\n    const classLevelPermissions = message.classLevelPermissions;\n    let currentParseObject = message.currentParseObject.toJSON();\n    const className = currentParseObject.className;\n    logger.verbose('ClassName: %s | ObjectId: %s', className, currentParseObject.id);\n    logger.verbose('Current client number : %d', this.clients.size);\n\n    const classSubscriptions = this.subscriptions.get(className);\n    if (typeof classSubscriptions === 'undefined') {\n      logger.debug('Can not find subscriptions under this class ' + className);\n      return;\n    }\n    for (const subscription of classSubscriptions.values()) {\n      const isOriginalSubscriptionMatched = this._matchesSubscription(\n        originalParseObject,\n        subscription\n      );\n      const isCurrentSubscriptionMatched = this._matchesSubscription(\n        currentParseObject,\n        subscription\n      );\n      for (const [clientId, requestIds] of _.entries(subscription.clientRequestIds)) {\n        const client = this.clients.get(clientId);\n        if (typeof client === 'undefined') {\n          continue;\n        }\n        requestIds.forEach(async requestId => {\n          // Set orignal ParseObject ACL checking promise, if the object does not match\n          // subscription, we do not need to check ACL\n          let originalACLCheckingPromise;\n          if (!isOriginalSubscriptionMatched) {\n            originalACLCheckingPromise = Promise.resolve(false);\n          } else {\n            let originalACL;\n            if (message.originalParseObject) {\n              originalACL = message.originalParseObject.getACL();\n            }\n            originalACLCheckingPromise = this._matchesACL(originalACL, client, requestId);\n          }\n          // Set current ParseObject ACL checking promise, if the object does not match\n          // subscription, we do not need to check ACL\n          let currentACLCheckingPromise;\n          let res = {};\n          if (!isCurrentSubscriptionMatched) {\n            currentACLCheckingPromise = Promise.resolve(false);\n          } else {\n            const currentACL = message.currentParseObject.getACL();\n            currentACLCheckingPromise = this._matchesACL(currentACL, client, requestId);\n          }\n          try {\n            const op = this._getCLPOperation(subscription.query);\n            await this._matchesCLP(\n              classLevelPermissions,\n              message.currentParseObject,\n              client,\n              requestId,\n              op\n            );\n            const [isOriginalMatched, isCurrentMatched] = await Promise.all([\n              originalACLCheckingPromise,\n              currentACLCheckingPromise,\n            ]);\n            logger.verbose(\n              'Original %j | Current %j | Match: %s, %s, %s, %s | Query: %s',\n              originalParseObject,\n              currentParseObject,\n              isOriginalSubscriptionMatched,\n              isCurrentSubscriptionMatched,\n              isOriginalMatched,\n              isCurrentMatched,\n              subscription.hash\n            );\n            // Decide event type\n            let type;\n            if (isOriginalMatched && isCurrentMatched) {\n              type = 'update';\n            } else if (isOriginalMatched && !isCurrentMatched) {\n              type = 'leave';\n            } else if (!isOriginalMatched && isCurrentMatched) {\n              if (originalParseObject) {\n                type = 'enter';\n              } else {\n                type = 'create';\n              }\n            } else {\n              return null;\n            }\n            const watchFieldsChanged = this._checkWatchFields(client, requestId, message);\n            if (!watchFieldsChanged && (type === 'update' || type === 'create')) {\n              return;\n            }\n            res = {\n              event: type,\n              sessionToken: client.sessionToken,\n              object: currentParseObject,\n              original: originalParseObject,\n              clients: this.clients.size,\n              subscriptions: this.subscriptions.size,\n              useMasterKey: client.hasMasterKey,\n              installationId: client.installationId,\n              sendEvent: true,\n            };\n            const trigger = getTrigger(className, 'afterEvent', Parse.applicationId);\n            if (trigger) {\n              if (res.object) {\n                res.object = Parse.Object.fromJSON(res.object);\n              }\n              if (res.original) {\n                res.original = Parse.Object.fromJSON(res.original);\n              }\n              const auth = await this.getAuthFromClient(client, requestId);\n              if (auth && auth.user) {\n                res.user = auth.user;\n              }\n              await runTrigger(trigger, `afterEvent.${className}`, res, auth);\n            }\n            if (!res.sendEvent) {\n              return;\n            }\n            if (res.object && typeof res.object.toJSON === 'function') {\n              currentParseObject = toJSONwithObjects(res.object, res.object.className || className);\n            }\n            if (res.original && typeof res.original.toJSON === 'function') {\n              originalParseObject = toJSONwithObjects(\n                res.original,\n                res.original.className || className\n              );\n            }\n            await this._filterSensitiveData(\n              classLevelPermissions,\n              res,\n              client,\n              requestId,\n              op,\n              subscription.query\n            );\n            const functionName = 'push' + res.event.charAt(0).toUpperCase() + res.event.slice(1);\n            if (client[functionName]) {\n              client[functionName](requestId, currentParseObject, originalParseObject);\n            }\n          } catch (e) {\n            const error = resolveError(e);\n            Client.pushError(client.parseWebSocket, error.code, error.message, false, requestId);\n            logger.error(\n              `Failed running afterLiveQueryEvent on class ${className} for event ${res.event} with session ${res.sessionToken} with:\\n Error: ` +\n                JSON.stringify(error)\n            );\n          }\n        });\n      }\n    }\n  }\n\n  _onConnect(parseWebsocket: any): void {\n    parseWebsocket.on('message', request => {\n      if (typeof request === 'string') {\n        try {\n          request = JSON.parse(request);\n        } catch (e) {\n          logger.error('unable to parse request', request, e);\n          return;\n        }\n      }\n      logger.verbose('Request: %j', request);\n\n      // Check whether this request is a valid request, return error directly if not\n      if (\n        !tv4.validate(request, RequestSchema['general']) ||\n        !tv4.validate(request, RequestSchema[request.op])\n      ) {\n        Client.pushError(parseWebsocket, 1, tv4.error.message);\n        logger.error('Connect message error %s', tv4.error.message);\n        return;\n      }\n\n      switch (request.op) {\n        case 'connect':\n          this._handleConnect(parseWebsocket, request);\n          break;\n        case 'subscribe':\n          this._handleSubscribe(parseWebsocket, request);\n          break;\n        case 'update':\n          this._handleUpdateSubscription(parseWebsocket, request);\n          break;\n        case 'unsubscribe':\n          this._handleUnsubscribe(parseWebsocket, request);\n          break;\n        default:\n          Client.pushError(parseWebsocket, 3, 'Get unknown operation');\n          logger.error('Get unknown operation', request.op);\n      }\n    });\n\n    parseWebsocket.on('disconnect', () => {\n      logger.info(`Client disconnect: ${parseWebsocket.clientId}`);\n      const clientId = parseWebsocket.clientId;\n      if (!this.clients.has(clientId)) {\n        runLiveQueryEventHandlers({\n          event: 'ws_disconnect_error',\n          clients: this.clients.size,\n          subscriptions: this.subscriptions.size,\n          error: `Unable to find client ${clientId}`,\n        });\n        logger.error(`Can not find client ${clientId} on disconnect`);\n        return;\n      }\n\n      // Delete client\n      const client = this.clients.get(clientId);\n      this.clients.delete(clientId);\n\n      // Delete client from subscriptions\n      for (const [requestId, subscriptionInfo] of _.entries(client.subscriptionInfos)) {\n        const subscription = subscriptionInfo.subscription;\n        subscription.deleteClientSubscription(clientId, requestId);\n\n        // If there is no client which is subscribing this subscription, remove it from subscriptions\n        const classSubscriptions = this.subscriptions.get(subscription.className);\n        if (!subscription.hasSubscribingClient()) {\n          classSubscriptions.delete(subscription.hash);\n        }\n        // If there is no subscriptions under this class, remove it from subscriptions\n        if (classSubscriptions.size === 0) {\n          this.subscriptions.delete(subscription.className);\n        }\n      }\n\n      logger.verbose('Current clients %d', this.clients.size);\n      logger.verbose('Current subscriptions %d', this.subscriptions.size);\n      runLiveQueryEventHandlers({\n        event: 'ws_disconnect',\n        clients: this.clients.size,\n        subscriptions: this.subscriptions.size,\n        useMasterKey: client.hasMasterKey,\n        installationId: client.installationId,\n        sessionToken: client.sessionToken,\n      });\n    });\n\n    runLiveQueryEventHandlers({\n      event: 'ws_connect',\n      clients: this.clients.size,\n      subscriptions: this.subscriptions.size,\n    });\n  }\n\n  _matchesSubscription(parseObject: any, subscription: any): boolean {\n    // Object is undefined or null, not match\n    if (!parseObject) {\n      return false;\n    }\n    return matchesQuery(deepcopy(parseObject), subscription.query);\n  }\n\n  async _clearCachedRoles(userId: string) {\n    try {\n      const validTokens = await new Parse.Query(Parse.Session)\n        .equalTo('user', Parse.User.createWithoutData(userId))\n        .find({ useMasterKey: true });\n      await Promise.all(\n        validTokens.map(async token => {\n          const sessionToken = token.get('sessionToken');\n          const authPromise = this.authCache.get(sessionToken);\n          if (!authPromise) {\n            return;\n          }\n          const [auth1, auth2] = await Promise.all([\n            authPromise,\n            getAuthForSessionToken({ cacheController: this.cacheController, sessionToken }),\n          ]);\n          auth1.auth?.clearRoleCache(sessionToken);\n          auth2.auth?.clearRoleCache(sessionToken);\n          this.authCache.delete(sessionToken);\n        })\n      );\n    } catch (e) {\n      logger.verbose(`Could not clear role cache. ${e}`);\n    }\n  }\n\n  getAuthForSessionToken(sessionToken: ?string): Promise<{ auth: ?Auth, userId: ?string }> {\n    if (!sessionToken) {\n      return Promise.resolve({});\n    }\n    const fromCache = this.authCache.get(sessionToken);\n    if (fromCache) {\n      return fromCache;\n    }\n    const authPromise = getAuthForSessionToken({\n      cacheController: this.cacheController,\n      sessionToken: sessionToken,\n    })\n      .then(auth => {\n        return { auth, userId: auth && auth.user && auth.user.id };\n      })\n      .catch(error => {\n        // There was an error with the session token\n        const result = {};\n        if (error && error.code === Parse.Error.INVALID_SESSION_TOKEN) {\n          result.error = error;\n          this.authCache.set(sessionToken, Promise.resolve(result), this.config.cacheTimeout);\n        } else {\n          this.authCache.delete(sessionToken);\n        }\n        return result;\n      });\n    this.authCache.set(sessionToken, authPromise);\n    return authPromise;\n  }\n\n  async _matchesCLP(\n    classLevelPermissions: ?any,\n    object: any,\n    client: any,\n    requestId: number,\n    op: string\n  ): any {\n    // try to match on user first, less expensive than with roles\n    const subscriptionInfo = client.getSubscriptionInfo(requestId);\n    const aclGroup = ['*'];\n    let userId;\n    if (typeof subscriptionInfo !== 'undefined') {\n      const { userId } = await this.getAuthForSessionToken(subscriptionInfo.sessionToken);\n      if (userId) {\n        aclGroup.push(userId);\n      }\n    }\n    try {\n      await SchemaController.validatePermission(\n        classLevelPermissions,\n        object.className,\n        aclGroup,\n        op\n      );\n      return true;\n    } catch (e) {\n      logger.verbose(`Failed matching CLP for ${object.id} ${userId} ${e}`);\n      return false;\n    }\n    // TODO: handle roles permissions\n    // Object.keys(classLevelPermissions).forEach((key) => {\n    //   const perm = classLevelPermissions[key];\n    //   Object.keys(perm).forEach((key) => {\n    //     if (key.indexOf('role'))\n    //   });\n    // })\n    // // it's rejected here, check the roles\n    // var rolesQuery = new Parse.Query(Parse.Role);\n    // rolesQuery.equalTo(\"users\", user);\n    // return rolesQuery.find({useMasterKey:true});\n  }\n\n  async _filterSensitiveData(\n    classLevelPermissions: ?any,\n    res: any,\n    client: any,\n    requestId: number,\n    op: string,\n    query: any\n  ) {\n    const subscriptionInfo = client.getSubscriptionInfo(requestId);\n    const aclGroup = ['*'];\n    let clientAuth;\n    if (typeof subscriptionInfo !== 'undefined') {\n      const { userId, auth } = await this.getAuthForSessionToken(subscriptionInfo.sessionToken);\n      if (userId) {\n        aclGroup.push(userId);\n      }\n      clientAuth = auth;\n    }\n    const filter = obj => {\n      if (!obj) {\n        return;\n      }\n      let protectedFields = classLevelPermissions?.protectedFields || [];\n      if (!client.hasMasterKey && !Array.isArray(protectedFields)) {\n        protectedFields = getDatabaseController(this.config).addProtectedFields(\n          classLevelPermissions,\n          res.object.className,\n          query,\n          aclGroup,\n          clientAuth\n        );\n      }\n      return DatabaseController.filterSensitiveData(\n        client.hasMasterKey,\n        false,\n        aclGroup,\n        clientAuth,\n        op,\n        classLevelPermissions,\n        res.object.className,\n        protectedFields,\n        obj,\n        query\n      );\n    };\n    res.object = filter(res.object);\n    res.original = filter(res.original);\n  }\n\n  _getCLPOperation(query: any) {\n    return typeof query === 'object' &&\n      Object.keys(query).length == 1 &&\n      typeof query.objectId === 'string'\n      ? 'get'\n      : 'find';\n  }\n\n  async _verifyACL(acl: any, token: string) {\n    if (!token) {\n      return false;\n    }\n\n    const { auth, userId } = await this.getAuthForSessionToken(token);\n\n    // Getting the session token failed\n    // This means that no additional auth is available\n    // At this point, just bail out as no additional visibility can be inferred.\n    if (!auth || !userId) {\n      return false;\n    }\n    const isSubscriptionSessionTokenMatched = acl.getReadAccess(userId);\n    if (isSubscriptionSessionTokenMatched) {\n      return true;\n    }\n\n    // Check if the user has any roles that match the ACL\n    return Promise.resolve()\n      .then(async () => {\n        // Resolve false right away if the acl doesn't have any roles\n        const acl_has_roles = Object.keys(acl.permissionsById).some(key => key.startsWith('role:'));\n        if (!acl_has_roles) {\n          return false;\n        }\n        const roleNames = await auth.getUserRoles();\n        // Finally, see if any of the user's roles allow them read access\n        for (const role of roleNames) {\n          // We use getReadAccess as `role` is in the form `role:roleName`\n          if (acl.getReadAccess(role)) {\n            return true;\n          }\n        }\n        return false;\n      })\n      .catch(() => {\n        return false;\n      });\n  }\n\n  async getAuthFromClient(client: any, requestId: number, sessionToken: string) {\n    const getSessionFromClient = () => {\n      const subscriptionInfo = client.getSubscriptionInfo(requestId);\n      if (typeof subscriptionInfo === 'undefined') {\n        return client.sessionToken;\n      }\n      return subscriptionInfo.sessionToken || client.sessionToken;\n    };\n    if (!sessionToken) {\n      sessionToken = getSessionFromClient();\n    }\n    if (!sessionToken) {\n      return;\n    }\n    const { auth } = await this.getAuthForSessionToken(sessionToken);\n    return auth;\n  }\n\n  _checkWatchFields(client: any, requestId: any, message: any) {\n    const subscriptionInfo = client.getSubscriptionInfo(requestId);\n    const watch = subscriptionInfo?.watch;\n    if (!watch) {\n      return true;\n    }\n    const object = message.currentParseObject;\n    const original = message.originalParseObject;\n    return watch.some(field => !isDeepStrictEqual(object.get(field), original?.get(field)));\n  }\n\n  async _matchesACL(acl: any, client: any, requestId: number): Promise<boolean> {\n    // Return true directly if ACL isn't present, ACL is public read, or client has master key\n    if (!acl || acl.getPublicReadAccess() || client.hasMasterKey) {\n      return true;\n    }\n    // Check subscription sessionToken matches ACL first\n    const subscriptionInfo = client.getSubscriptionInfo(requestId);\n    if (typeof subscriptionInfo === 'undefined') {\n      return false;\n    }\n\n    const subscriptionToken = subscriptionInfo.sessionToken;\n    const clientSessionToken = client.sessionToken;\n\n    if (await this._verifyACL(acl, subscriptionToken)) {\n      return true;\n    }\n\n    if (await this._verifyACL(acl, clientSessionToken)) {\n      return true;\n    }\n\n    return false;\n  }\n\n  async _handleConnect(parseWebsocket: any, request: any): any {\n    if (!this._validateKeys(request, this.keyPairs)) {\n      Client.pushError(parseWebsocket, 4, 'Key in request is not valid');\n      logger.error('Key in request is not valid');\n      return;\n    }\n    const hasMasterKey = this._hasMasterKey(request, this.keyPairs);\n    const clientId = uuidv4();\n    const client = new Client(\n      clientId,\n      parseWebsocket,\n      hasMasterKey,\n      request.sessionToken,\n      request.installationId\n    );\n    try {\n      const req = {\n        client,\n        event: 'connect',\n        clients: this.clients.size,\n        subscriptions: this.subscriptions.size,\n        sessionToken: request.sessionToken,\n        useMasterKey: client.hasMasterKey,\n        installationId: request.installationId,\n      };\n      const trigger = getTrigger('@Connect', 'beforeConnect', Parse.applicationId);\n      if (trigger) {\n        const auth = await this.getAuthFromClient(client, request.requestId, req.sessionToken);\n        if (auth && auth.user) {\n          req.user = auth.user;\n        }\n        await runTrigger(trigger, `beforeConnect.@Connect`, req, auth);\n      }\n      parseWebsocket.clientId = clientId;\n      this.clients.set(parseWebsocket.clientId, client);\n      logger.info(`Create new client: ${parseWebsocket.clientId}`);\n      client.pushConnect();\n      runLiveQueryEventHandlers(req);\n    } catch (e) {\n      const error = resolveError(e);\n      Client.pushError(parseWebsocket, error.code, error.message, false);\n      logger.error(\n        `Failed running beforeConnect for session ${request.sessionToken} with:\\n Error: ` +\n          JSON.stringify(error)\n      );\n    }\n  }\n\n  _hasMasterKey(request: any, validKeyPairs: any): boolean {\n    if (!validKeyPairs || validKeyPairs.size == 0 || !validKeyPairs.has('masterKey')) {\n      return false;\n    }\n    if (!request || !Object.prototype.hasOwnProperty.call(request, 'masterKey')) {\n      return false;\n    }\n    return request.masterKey === validKeyPairs.get('masterKey');\n  }\n\n  _validateKeys(request: any, validKeyPairs: any): boolean {\n    if (!validKeyPairs || validKeyPairs.size == 0) {\n      return true;\n    }\n    let isValid = false;\n    for (const [key, secret] of validKeyPairs) {\n      if (!request[key] || request[key] !== secret) {\n        continue;\n      }\n      isValid = true;\n      break;\n    }\n    return isValid;\n  }\n\n  async _handleSubscribe(parseWebsocket: any, request: any): any {\n    // If we can not find this client, return error to client\n    if (!Object.prototype.hasOwnProperty.call(parseWebsocket, 'clientId')) {\n      Client.pushError(\n        parseWebsocket,\n        2,\n        'Can not find this client, make sure you connect to server before subscribing'\n      );\n      logger.error('Can not find this client, make sure you connect to server before subscribing');\n      return;\n    }\n    const client = this.clients.get(parseWebsocket.clientId);\n    const className = request.query.className;\n    let authCalled = false;\n    try {\n      const trigger = getTrigger(className, 'beforeSubscribe', Parse.applicationId);\n      if (trigger) {\n        const auth = await this.getAuthFromClient(client, request.requestId, request.sessionToken);\n        authCalled = true;\n        if (auth && auth.user) {\n          request.user = auth.user;\n        }\n\n        const parseQuery = new Parse.Query(className);\n        parseQuery.withJSON(request.query);\n        request.query = parseQuery;\n        await runTrigger(trigger, `beforeSubscribe.${className}`, request, auth);\n\n        const query = request.query.toJSON();\n        request.query = query;\n      }\n\n      if (className === '_Session') {\n        if (!authCalled) {\n          const auth = await this.getAuthFromClient(\n            client,\n            request.requestId,\n            request.sessionToken\n          );\n          if (auth && auth.user) {\n            request.user = auth.user;\n          }\n        }\n        if (request.user) {\n          request.query.where.user = request.user.toPointer();\n        } else if (!request.master) {\n          Client.pushError(\n            parseWebsocket,\n            Parse.Error.INVALID_SESSION_TOKEN,\n            'Invalid session token',\n            false,\n            request.requestId\n          );\n          return;\n        }\n      }\n      // Get subscription from subscriptions, create one if necessary\n      const subscriptionHash = queryHash(request.query);\n      // Add className to subscriptions if necessary\n\n      if (!this.subscriptions.has(className)) {\n        this.subscriptions.set(className, new Map());\n      }\n      const classSubscriptions = this.subscriptions.get(className);\n      let subscription;\n      if (classSubscriptions.has(subscriptionHash)) {\n        subscription = classSubscriptions.get(subscriptionHash);\n      } else {\n        subscription = new Subscription(className, request.query.where, subscriptionHash);\n        classSubscriptions.set(subscriptionHash, subscription);\n      }\n\n      // Add subscriptionInfo to client\n      const subscriptionInfo = {\n        subscription: subscription,\n      };\n      // Add selected fields, sessionToken and installationId for this subscription if necessary\n      if (request.query.keys) {\n        subscriptionInfo.keys = Array.isArray(request.query.keys)\n          ? request.query.keys\n          : request.query.keys.split(',');\n      }\n      if (request.query.watch) {\n        subscriptionInfo.watch = request.query.watch;\n      }\n      if (request.sessionToken) {\n        subscriptionInfo.sessionToken = request.sessionToken;\n      }\n      client.addSubscriptionInfo(request.requestId, subscriptionInfo);\n\n      // Add clientId to subscription\n      subscription.addClientSubscription(parseWebsocket.clientId, request.requestId);\n\n      client.pushSubscribe(request.requestId);\n\n      logger.verbose(\n        `Create client ${parseWebsocket.clientId} new subscription: ${request.requestId}`\n      );\n      logger.verbose('Current client number: %d', this.clients.size);\n      runLiveQueryEventHandlers({\n        client,\n        event: 'subscribe',\n        clients: this.clients.size,\n        subscriptions: this.subscriptions.size,\n        sessionToken: request.sessionToken,\n        useMasterKey: client.hasMasterKey,\n        installationId: client.installationId,\n      });\n    } catch (e) {\n      const error = resolveError(e);\n      Client.pushError(parseWebsocket, error.code, error.message, false, request.requestId);\n      logger.error(\n        `Failed running beforeSubscribe on ${className} for session ${request.sessionToken} with:\\n Error: ` +\n          JSON.stringify(error)\n      );\n    }\n  }\n\n  _handleUpdateSubscription(parseWebsocket: any, request: any): any {\n    this._handleUnsubscribe(parseWebsocket, request, false);\n    this._handleSubscribe(parseWebsocket, request);\n  }\n\n  _handleUnsubscribe(parseWebsocket: any, request: any, notifyClient: boolean = true): any {\n    // If we can not find this client, return error to client\n    if (!Object.prototype.hasOwnProperty.call(parseWebsocket, 'clientId')) {\n      Client.pushError(\n        parseWebsocket,\n        2,\n        'Can not find this client, make sure you connect to server before unsubscribing'\n      );\n      logger.error(\n        'Can not find this client, make sure you connect to server before unsubscribing'\n      );\n      return;\n    }\n    const requestId = request.requestId;\n    const client = this.clients.get(parseWebsocket.clientId);\n    if (typeof client === 'undefined') {\n      Client.pushError(\n        parseWebsocket,\n        2,\n        'Cannot find client with clientId ' +\n          parseWebsocket.clientId +\n          '. Make sure you connect to live query server before unsubscribing.'\n      );\n      logger.error('Can not find this client ' + parseWebsocket.clientId);\n      return;\n    }\n\n    const subscriptionInfo = client.getSubscriptionInfo(requestId);\n    if (typeof subscriptionInfo === 'undefined') {\n      Client.pushError(\n        parseWebsocket,\n        2,\n        'Cannot find subscription with clientId ' +\n          parseWebsocket.clientId +\n          ' subscriptionId ' +\n          requestId +\n          '. Make sure you subscribe to live query server before unsubscribing.'\n      );\n      logger.error(\n        'Can not find subscription with clientId ' +\n          parseWebsocket.clientId +\n          ' subscriptionId ' +\n          requestId\n      );\n      return;\n    }\n\n    // Remove subscription from client\n    client.deleteSubscriptionInfo(requestId);\n    // Remove client from subscription\n    const subscription = subscriptionInfo.subscription;\n    const className = subscription.className;\n    subscription.deleteClientSubscription(parseWebsocket.clientId, requestId);\n    // If there is no client which is subscribing this subscription, remove it from subscriptions\n    const classSubscriptions = this.subscriptions.get(className);\n    if (!subscription.hasSubscribingClient()) {\n      classSubscriptions.delete(subscription.hash);\n    }\n    // If there is no subscriptions under this class, remove it from subscriptions\n    if (classSubscriptions.size === 0) {\n      this.subscriptions.delete(className);\n    }\n    runLiveQueryEventHandlers({\n      client,\n      event: 'unsubscribe',\n      clients: this.clients.size,\n      subscriptions: this.subscriptions.size,\n      sessionToken: subscriptionInfo.sessionToken,\n      useMasterKey: client.hasMasterKey,\n      installationId: client.installationId,\n    });\n\n    if (!notifyClient) {\n      return;\n    }\n\n    client.pushUnsubscribe(request.requestId);\n\n    logger.verbose(\n      `Delete client: ${parseWebsocket.clientId} | subscription: ${request.requestId}`\n    );\n  }\n}\n\nexport { ParseLiveQueryServer };\n"],"mappings":"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","ignoreList":[]}