123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845 |
- "use strict";
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- exports.ParseLiveQueryServer = void 0;
- var _tv = _interopRequireDefault(require("tv4"));
- var _node = _interopRequireDefault(require("parse/node"));
- var _Subscription = require("./Subscription");
- var _Client = require("./Client");
- var _ParseWebSocketServer = require("./ParseWebSocketServer");
- var _logger = _interopRequireDefault(require("../logger"));
- var _RequestSchema = _interopRequireDefault(require("./RequestSchema"));
- var _QueryTools = require("./QueryTools");
- var _ParsePubSub = require("./ParsePubSub");
- var _SchemaController = _interopRequireDefault(require("../Controllers/SchemaController"));
- var _lodash = _interopRequireDefault(require("lodash"));
- var _uuid = require("uuid");
- var _triggers = require("../triggers");
- var _Auth = require("../Auth");
- var _Controllers = require("../Controllers");
- var _lruCache = require("lru-cache");
- var _UsersRouter = _interopRequireDefault(require("../Routers/UsersRouter"));
- var _DatabaseController = _interopRequireDefault(require("../Controllers/DatabaseController"));
- var _util = require("util");
- var _deepcopy = _interopRequireDefault(require("deepcopy"));
- function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
- class ParseLiveQueryServer {
- // className -> (queryHash -> subscription)
- // The subscriber we use to get object update from publisher
- constructor(server, config = {}, parseServerConfig = {}) {
- this.server = server;
- this.clients = new Map();
- this.subscriptions = new Map();
- this.config = config;
- config.appId = config.appId || _node.default.applicationId;
- config.masterKey = config.masterKey || _node.default.masterKey;
- // Store keys, convert obj to map
- const keyPairs = config.keyPairs || {};
- this.keyPairs = new Map();
- for (const key of Object.keys(keyPairs)) {
- this.keyPairs.set(key, keyPairs[key]);
- }
- _logger.default.verbose('Support key pairs', this.keyPairs);
- // Initialize Parse
- _node.default.Object.disableSingleInstance();
- const serverURL = config.serverURL || _node.default.serverURL;
- _node.default.serverURL = serverURL;
- _node.default.initialize(config.appId, _node.default.javaScriptKey, config.masterKey);
- // The cache controller is a proper cache controller
- // with access to User and Roles
- this.cacheController = (0, _Controllers.getCacheController)(parseServerConfig);
- config.cacheTimeout = config.cacheTimeout || 5 * 1000; // 5s
- // This auth cache stores the promises for each auth resolution.
- // The main benefit is to be able to reuse the same user / session token resolution.
- this.authCache = new _lruCache.LRUCache({
- max: 500,
- // 500 concurrent
- ttl: config.cacheTimeout
- });
- // Initialize websocket server
- this.parseWebSocketServer = new _ParseWebSocketServer.ParseWebSocketServer(server, parseWebsocket => this._onConnect(parseWebsocket), config);
- this.subscriber = _ParsePubSub.ParsePubSub.createSubscriber(config);
- if (!this.subscriber.connect) {
- this.connect();
- }
- }
- async connect() {
- if (this.subscriber.isOpen) {
- return;
- }
- if (typeof this.subscriber.connect === 'function') {
- await Promise.resolve(this.subscriber.connect());
- } else {
- this.subscriber.isOpen = true;
- }
- this._createSubscribers();
- }
- async shutdown() {
- if (this.subscriber.isOpen) {
- var _this$subscriber$clos, _this$subscriber;
- 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)]);
- }
- this.subscriber.isOpen = false;
- }
- _createSubscribers() {
- const messageRecieved = (channel, messageStr) => {
- _logger.default.verbose('Subscribe message %j', messageStr);
- let message;
- try {
- message = JSON.parse(messageStr);
- } catch (e) {
- _logger.default.error('unable to parse message', messageStr, e);
- return;
- }
- if (channel === _node.default.applicationId + 'clearCache') {
- this._clearCachedRoles(message.userId);
- return;
- }
- this._inflateParseObject(message);
- if (channel === _node.default.applicationId + 'afterSave') {
- this._onAfterSave(message);
- } else if (channel === _node.default.applicationId + 'afterDelete') {
- this._onAfterDelete(message);
- } else {
- _logger.default.error('Get message %s from unknown channel %j', message, channel);
- }
- };
- this.subscriber.on('message', (channel, messageStr) => messageRecieved(channel, messageStr));
- for (const field of ['afterSave', 'afterDelete', 'clearCache']) {
- const channel = `${_node.default.applicationId}${field}`;
- this.subscriber.subscribe(channel, messageStr => messageRecieved(channel, messageStr));
- }
- }
- // Message is the JSON object from publisher. Message.currentParseObject is the ParseObject JSON after changes.
- // Message.originalParseObject is the original ParseObject JSON.
- _inflateParseObject(message) {
- // Inflate merged object
- const currentParseObject = message.currentParseObject;
- _UsersRouter.default.removeHiddenProperties(currentParseObject);
- let className = currentParseObject.className;
- let parseObject = new _node.default.Object(className);
- parseObject._finishFetch(currentParseObject);
- message.currentParseObject = parseObject;
- // Inflate original object
- const originalParseObject = message.originalParseObject;
- if (originalParseObject) {
- _UsersRouter.default.removeHiddenProperties(originalParseObject);
- className = originalParseObject.className;
- parseObject = new _node.default.Object(className);
- parseObject._finishFetch(originalParseObject);
- message.originalParseObject = parseObject;
- }
- }
- // Message is the JSON object from publisher after inflated. Message.currentParseObject is the ParseObject after changes.
- // Message.originalParseObject is the original ParseObject.
- async _onAfterDelete(message) {
- _logger.default.verbose(_node.default.applicationId + 'afterDelete is triggered');
- let deletedParseObject = message.currentParseObject.toJSON();
- const classLevelPermissions = message.classLevelPermissions;
- const className = deletedParseObject.className;
- _logger.default.verbose('ClassName: %j | ObjectId: %s', className, deletedParseObject.id);
- _logger.default.verbose('Current client number : %d', this.clients.size);
- const classSubscriptions = this.subscriptions.get(className);
- if (typeof classSubscriptions === 'undefined') {
- _logger.default.debug('Can not find subscriptions under this class ' + className);
- return;
- }
- for (const subscription of classSubscriptions.values()) {
- const isSubscriptionMatched = this._matchesSubscription(deletedParseObject, subscription);
- if (!isSubscriptionMatched) {
- continue;
- }
- for (const [clientId, requestIds] of _lodash.default.entries(subscription.clientRequestIds)) {
- const client = this.clients.get(clientId);
- if (typeof client === 'undefined') {
- continue;
- }
- requestIds.forEach(async requestId => {
- const acl = message.currentParseObject.getACL();
- // Check CLP
- const op = this._getCLPOperation(subscription.query);
- let res = {};
- try {
- await this._matchesCLP(classLevelPermissions, message.currentParseObject, client, requestId, op);
- const isMatched = await this._matchesACL(acl, client, requestId);
- if (!isMatched) {
- return null;
- }
- res = {
- event: 'delete',
- sessionToken: client.sessionToken,
- object: deletedParseObject,
- clients: this.clients.size,
- subscriptions: this.subscriptions.size,
- useMasterKey: client.hasMasterKey,
- installationId: client.installationId,
- sendEvent: true
- };
- const trigger = (0, _triggers.getTrigger)(className, 'afterEvent', _node.default.applicationId);
- if (trigger) {
- const auth = await this.getAuthFromClient(client, requestId);
- if (auth && auth.user) {
- res.user = auth.user;
- }
- if (res.object) {
- res.object = _node.default.Object.fromJSON(res.object);
- }
- await (0, _triggers.runTrigger)(trigger, `afterEvent.${className}`, res, auth);
- }
- if (!res.sendEvent) {
- return;
- }
- if (res.object && typeof res.object.toJSON === 'function') {
- deletedParseObject = (0, _triggers.toJSONwithObjects)(res.object, res.object.className || className);
- }
- await this._filterSensitiveData(classLevelPermissions, res, client, requestId, op, subscription.query);
- client.pushDelete(requestId, deletedParseObject);
- } catch (e) {
- const error = (0, _triggers.resolveError)(e);
- _Client.Client.pushError(client.parseWebSocket, error.code, error.message, false, requestId);
- _logger.default.error(`Failed running afterLiveQueryEvent on class ${className} for event ${res.event} with session ${res.sessionToken} with:\n Error: ` + JSON.stringify(error));
- }
- });
- }
- }
- }
- // Message is the JSON object from publisher after inflated. Message.currentParseObject is the ParseObject after changes.
- // Message.originalParseObject is the original ParseObject.
- async _onAfterSave(message) {
- _logger.default.verbose(_node.default.applicationId + 'afterSave is triggered');
- let originalParseObject = null;
- if (message.originalParseObject) {
- originalParseObject = message.originalParseObject.toJSON();
- }
- const classLevelPermissions = message.classLevelPermissions;
- let currentParseObject = message.currentParseObject.toJSON();
- const className = currentParseObject.className;
- _logger.default.verbose('ClassName: %s | ObjectId: %s', className, currentParseObject.id);
- _logger.default.verbose('Current client number : %d', this.clients.size);
- const classSubscriptions = this.subscriptions.get(className);
- if (typeof classSubscriptions === 'undefined') {
- _logger.default.debug('Can not find subscriptions under this class ' + className);
- return;
- }
- for (const subscription of classSubscriptions.values()) {
- const isOriginalSubscriptionMatched = this._matchesSubscription(originalParseObject, subscription);
- const isCurrentSubscriptionMatched = this._matchesSubscription(currentParseObject, subscription);
- for (const [clientId, requestIds] of _lodash.default.entries(subscription.clientRequestIds)) {
- const client = this.clients.get(clientId);
- if (typeof client === 'undefined') {
- continue;
- }
- requestIds.forEach(async requestId => {
- // Set orignal ParseObject ACL checking promise, if the object does not match
- // subscription, we do not need to check ACL
- let originalACLCheckingPromise;
- if (!isOriginalSubscriptionMatched) {
- originalACLCheckingPromise = Promise.resolve(false);
- } else {
- let originalACL;
- if (message.originalParseObject) {
- originalACL = message.originalParseObject.getACL();
- }
- originalACLCheckingPromise = this._matchesACL(originalACL, client, requestId);
- }
- // Set current ParseObject ACL checking promise, if the object does not match
- // subscription, we do not need to check ACL
- let currentACLCheckingPromise;
- let res = {};
- if (!isCurrentSubscriptionMatched) {
- currentACLCheckingPromise = Promise.resolve(false);
- } else {
- const currentACL = message.currentParseObject.getACL();
- currentACLCheckingPromise = this._matchesACL(currentACL, client, requestId);
- }
- try {
- const op = this._getCLPOperation(subscription.query);
- await this._matchesCLP(classLevelPermissions, message.currentParseObject, client, requestId, op);
- const [isOriginalMatched, isCurrentMatched] = await Promise.all([originalACLCheckingPromise, currentACLCheckingPromise]);
- _logger.default.verbose('Original %j | Current %j | Match: %s, %s, %s, %s | Query: %s', originalParseObject, currentParseObject, isOriginalSubscriptionMatched, isCurrentSubscriptionMatched, isOriginalMatched, isCurrentMatched, subscription.hash);
- // Decide event type
- let type;
- if (isOriginalMatched && isCurrentMatched) {
- type = 'update';
- } else if (isOriginalMatched && !isCurrentMatched) {
- type = 'leave';
- } else if (!isOriginalMatched && isCurrentMatched) {
- if (originalParseObject) {
- type = 'enter';
- } else {
- type = 'create';
- }
- } else {
- return null;
- }
- const watchFieldsChanged = this._checkWatchFields(client, requestId, message);
- if (!watchFieldsChanged && (type === 'update' || type === 'create')) {
- return;
- }
- res = {
- event: type,
- sessionToken: client.sessionToken,
- object: currentParseObject,
- original: originalParseObject,
- clients: this.clients.size,
- subscriptions: this.subscriptions.size,
- useMasterKey: client.hasMasterKey,
- installationId: client.installationId,
- sendEvent: true
- };
- const trigger = (0, _triggers.getTrigger)(className, 'afterEvent', _node.default.applicationId);
- if (trigger) {
- if (res.object) {
- res.object = _node.default.Object.fromJSON(res.object);
- }
- if (res.original) {
- res.original = _node.default.Object.fromJSON(res.original);
- }
- const auth = await this.getAuthFromClient(client, requestId);
- if (auth && auth.user) {
- res.user = auth.user;
- }
- await (0, _triggers.runTrigger)(trigger, `afterEvent.${className}`, res, auth);
- }
- if (!res.sendEvent) {
- return;
- }
- if (res.object && typeof res.object.toJSON === 'function') {
- currentParseObject = (0, _triggers.toJSONwithObjects)(res.object, res.object.className || className);
- }
- if (res.original && typeof res.original.toJSON === 'function') {
- originalParseObject = (0, _triggers.toJSONwithObjects)(res.original, res.original.className || className);
- }
- await this._filterSensitiveData(classLevelPermissions, res, client, requestId, op, subscription.query);
- const functionName = 'push' + res.event.charAt(0).toUpperCase() + res.event.slice(1);
- if (client[functionName]) {
- client[functionName](requestId, currentParseObject, originalParseObject);
- }
- } catch (e) {
- const error = (0, _triggers.resolveError)(e);
- _Client.Client.pushError(client.parseWebSocket, error.code, error.message, false, requestId);
- _logger.default.error(`Failed running afterLiveQueryEvent on class ${className} for event ${res.event} with session ${res.sessionToken} with:\n Error: ` + JSON.stringify(error));
- }
- });
- }
- }
- }
- _onConnect(parseWebsocket) {
- parseWebsocket.on('message', request => {
- if (typeof request === 'string') {
- try {
- request = JSON.parse(request);
- } catch (e) {
- _logger.default.error('unable to parse request', request, e);
- return;
- }
- }
- _logger.default.verbose('Request: %j', request);
- // Check whether this request is a valid request, return error directly if not
- if (!_tv.default.validate(request, _RequestSchema.default['general']) || !_tv.default.validate(request, _RequestSchema.default[request.op])) {
- _Client.Client.pushError(parseWebsocket, 1, _tv.default.error.message);
- _logger.default.error('Connect message error %s', _tv.default.error.message);
- return;
- }
- switch (request.op) {
- case 'connect':
- this._handleConnect(parseWebsocket, request);
- break;
- case 'subscribe':
- this._handleSubscribe(parseWebsocket, request);
- break;
- case 'update':
- this._handleUpdateSubscription(parseWebsocket, request);
- break;
- case 'unsubscribe':
- this._handleUnsubscribe(parseWebsocket, request);
- break;
- default:
- _Client.Client.pushError(parseWebsocket, 3, 'Get unknown operation');
- _logger.default.error('Get unknown operation', request.op);
- }
- });
- parseWebsocket.on('disconnect', () => {
- _logger.default.info(`Client disconnect: ${parseWebsocket.clientId}`);
- const clientId = parseWebsocket.clientId;
- if (!this.clients.has(clientId)) {
- (0, _triggers.runLiveQueryEventHandlers)({
- event: 'ws_disconnect_error',
- clients: this.clients.size,
- subscriptions: this.subscriptions.size,
- error: `Unable to find client ${clientId}`
- });
- _logger.default.error(`Can not find client ${clientId} on disconnect`);
- return;
- }
- // Delete client
- const client = this.clients.get(clientId);
- this.clients.delete(clientId);
- // Delete client from subscriptions
- for (const [requestId, subscriptionInfo] of _lodash.default.entries(client.subscriptionInfos)) {
- const subscription = subscriptionInfo.subscription;
- subscription.deleteClientSubscription(clientId, requestId);
- // If there is no client which is subscribing this subscription, remove it from subscriptions
- const classSubscriptions = this.subscriptions.get(subscription.className);
- if (!subscription.hasSubscribingClient()) {
- classSubscriptions.delete(subscription.hash);
- }
- // If there is no subscriptions under this class, remove it from subscriptions
- if (classSubscriptions.size === 0) {
- this.subscriptions.delete(subscription.className);
- }
- }
- _logger.default.verbose('Current clients %d', this.clients.size);
- _logger.default.verbose('Current subscriptions %d', this.subscriptions.size);
- (0, _triggers.runLiveQueryEventHandlers)({
- event: 'ws_disconnect',
- clients: this.clients.size,
- subscriptions: this.subscriptions.size,
- useMasterKey: client.hasMasterKey,
- installationId: client.installationId,
- sessionToken: client.sessionToken
- });
- });
- (0, _triggers.runLiveQueryEventHandlers)({
- event: 'ws_connect',
- clients: this.clients.size,
- subscriptions: this.subscriptions.size
- });
- }
- _matchesSubscription(parseObject, subscription) {
- // Object is undefined or null, not match
- if (!parseObject) {
- return false;
- }
- return (0, _QueryTools.matchesQuery)((0, _deepcopy.default)(parseObject), subscription.query);
- }
- async _clearCachedRoles(userId) {
- try {
- const validTokens = await new _node.default.Query(_node.default.Session).equalTo('user', _node.default.User.createWithoutData(userId)).find({
- useMasterKey: true
- });
- await Promise.all(validTokens.map(async token => {
- var _auth1$auth, _auth2$auth;
- const sessionToken = token.get('sessionToken');
- const authPromise = this.authCache.get(sessionToken);
- if (!authPromise) {
- return;
- }
- const [auth1, auth2] = await Promise.all([authPromise, (0, _Auth.getAuthForSessionToken)({
- cacheController: this.cacheController,
- sessionToken
- })]);
- (_auth1$auth = auth1.auth) === null || _auth1$auth === void 0 || _auth1$auth.clearRoleCache(sessionToken);
- (_auth2$auth = auth2.auth) === null || _auth2$auth === void 0 || _auth2$auth.clearRoleCache(sessionToken);
- this.authCache.delete(sessionToken);
- }));
- } catch (e) {
- _logger.default.verbose(`Could not clear role cache. ${e}`);
- }
- }
- getAuthForSessionToken(sessionToken) {
- if (!sessionToken) {
- return Promise.resolve({});
- }
- const fromCache = this.authCache.get(sessionToken);
- if (fromCache) {
- return fromCache;
- }
- const authPromise = (0, _Auth.getAuthForSessionToken)({
- cacheController: this.cacheController,
- sessionToken: sessionToken
- }).then(auth => {
- return {
- auth,
- userId: auth && auth.user && auth.user.id
- };
- }).catch(error => {
- // There was an error with the session token
- const result = {};
- if (error && error.code === _node.default.Error.INVALID_SESSION_TOKEN) {
- result.error = error;
- this.authCache.set(sessionToken, Promise.resolve(result), this.config.cacheTimeout);
- } else {
- this.authCache.delete(sessionToken);
- }
- return result;
- });
- this.authCache.set(sessionToken, authPromise);
- return authPromise;
- }
- async _matchesCLP(classLevelPermissions, object, client, requestId, op) {
- // try to match on user first, less expensive than with roles
- const subscriptionInfo = client.getSubscriptionInfo(requestId);
- const aclGroup = ['*'];
- let userId;
- if (typeof subscriptionInfo !== 'undefined') {
- const {
- userId
- } = await this.getAuthForSessionToken(subscriptionInfo.sessionToken);
- if (userId) {
- aclGroup.push(userId);
- }
- }
- try {
- await _SchemaController.default.validatePermission(classLevelPermissions, object.className, aclGroup, op);
- return true;
- } catch (e) {
- _logger.default.verbose(`Failed matching CLP for ${object.id} ${userId} ${e}`);
- return false;
- }
- // TODO: handle roles permissions
- // Object.keys(classLevelPermissions).forEach((key) => {
- // const perm = classLevelPermissions[key];
- // Object.keys(perm).forEach((key) => {
- // if (key.indexOf('role'))
- // });
- // })
- // // it's rejected here, check the roles
- // var rolesQuery = new Parse.Query(Parse.Role);
- // rolesQuery.equalTo("users", user);
- // return rolesQuery.find({useMasterKey:true});
- }
- async _filterSensitiveData(classLevelPermissions, res, client, requestId, op, query) {
- const subscriptionInfo = client.getSubscriptionInfo(requestId);
- const aclGroup = ['*'];
- let clientAuth;
- if (typeof subscriptionInfo !== 'undefined') {
- const {
- userId,
- auth
- } = await this.getAuthForSessionToken(subscriptionInfo.sessionToken);
- if (userId) {
- aclGroup.push(userId);
- }
- clientAuth = auth;
- }
- const filter = obj => {
- if (!obj) {
- return;
- }
- let protectedFields = (classLevelPermissions === null || classLevelPermissions === void 0 ? void 0 : classLevelPermissions.protectedFields) || [];
- if (!client.hasMasterKey && !Array.isArray(protectedFields)) {
- protectedFields = (0, _Controllers.getDatabaseController)(this.config).addProtectedFields(classLevelPermissions, res.object.className, query, aclGroup, clientAuth);
- }
- return _DatabaseController.default.filterSensitiveData(client.hasMasterKey, false, aclGroup, clientAuth, op, classLevelPermissions, res.object.className, protectedFields, obj, query);
- };
- res.object = filter(res.object);
- res.original = filter(res.original);
- }
- _getCLPOperation(query) {
- return typeof query === 'object' && Object.keys(query).length == 1 && typeof query.objectId === 'string' ? 'get' : 'find';
- }
- async _verifyACL(acl, token) {
- if (!token) {
- return false;
- }
- const {
- auth,
- userId
- } = await this.getAuthForSessionToken(token);
- // Getting the session token failed
- // This means that no additional auth is available
- // At this point, just bail out as no additional visibility can be inferred.
- if (!auth || !userId) {
- return false;
- }
- const isSubscriptionSessionTokenMatched = acl.getReadAccess(userId);
- if (isSubscriptionSessionTokenMatched) {
- return true;
- }
- // Check if the user has any roles that match the ACL
- return Promise.resolve().then(async () => {
- // Resolve false right away if the acl doesn't have any roles
- const acl_has_roles = Object.keys(acl.permissionsById).some(key => key.startsWith('role:'));
- if (!acl_has_roles) {
- return false;
- }
- const roleNames = await auth.getUserRoles();
- // Finally, see if any of the user's roles allow them read access
- for (const role of roleNames) {
- // We use getReadAccess as `role` is in the form `role:roleName`
- if (acl.getReadAccess(role)) {
- return true;
- }
- }
- return false;
- }).catch(() => {
- return false;
- });
- }
- async getAuthFromClient(client, requestId, sessionToken) {
- const getSessionFromClient = () => {
- const subscriptionInfo = client.getSubscriptionInfo(requestId);
- if (typeof subscriptionInfo === 'undefined') {
- return client.sessionToken;
- }
- return subscriptionInfo.sessionToken || client.sessionToken;
- };
- if (!sessionToken) {
- sessionToken = getSessionFromClient();
- }
- if (!sessionToken) {
- return;
- }
- const {
- auth
- } = await this.getAuthForSessionToken(sessionToken);
- return auth;
- }
- _checkWatchFields(client, requestId, message) {
- const subscriptionInfo = client.getSubscriptionInfo(requestId);
- const watch = subscriptionInfo === null || subscriptionInfo === void 0 ? void 0 : subscriptionInfo.watch;
- if (!watch) {
- return true;
- }
- const object = message.currentParseObject;
- const original = message.originalParseObject;
- return watch.some(field => !(0, _util.isDeepStrictEqual)(object.get(field), original === null || original === void 0 ? void 0 : original.get(field)));
- }
- async _matchesACL(acl, client, requestId) {
- // Return true directly if ACL isn't present, ACL is public read, or client has master key
- if (!acl || acl.getPublicReadAccess() || client.hasMasterKey) {
- return true;
- }
- // Check subscription sessionToken matches ACL first
- const subscriptionInfo = client.getSubscriptionInfo(requestId);
- if (typeof subscriptionInfo === 'undefined') {
- return false;
- }
- const subscriptionToken = subscriptionInfo.sessionToken;
- const clientSessionToken = client.sessionToken;
- if (await this._verifyACL(acl, subscriptionToken)) {
- return true;
- }
- if (await this._verifyACL(acl, clientSessionToken)) {
- return true;
- }
- return false;
- }
- async _handleConnect(parseWebsocket, request) {
- if (!this._validateKeys(request, this.keyPairs)) {
- _Client.Client.pushError(parseWebsocket, 4, 'Key in request is not valid');
- _logger.default.error('Key in request is not valid');
- return;
- }
- const hasMasterKey = this._hasMasterKey(request, this.keyPairs);
- const clientId = (0, _uuid.v4)();
- const client = new _Client.Client(clientId, parseWebsocket, hasMasterKey, request.sessionToken, request.installationId);
- try {
- const req = {
- client,
- event: 'connect',
- clients: this.clients.size,
- subscriptions: this.subscriptions.size,
- sessionToken: request.sessionToken,
- useMasterKey: client.hasMasterKey,
- installationId: request.installationId
- };
- const trigger = (0, _triggers.getTrigger)('@Connect', 'beforeConnect', _node.default.applicationId);
- if (trigger) {
- const auth = await this.getAuthFromClient(client, request.requestId, req.sessionToken);
- if (auth && auth.user) {
- req.user = auth.user;
- }
- await (0, _triggers.runTrigger)(trigger, `beforeConnect.@Connect`, req, auth);
- }
- parseWebsocket.clientId = clientId;
- this.clients.set(parseWebsocket.clientId, client);
- _logger.default.info(`Create new client: ${parseWebsocket.clientId}`);
- client.pushConnect();
- (0, _triggers.runLiveQueryEventHandlers)(req);
- } catch (e) {
- const error = (0, _triggers.resolveError)(e);
- _Client.Client.pushError(parseWebsocket, error.code, error.message, false);
- _logger.default.error(`Failed running beforeConnect for session ${request.sessionToken} with:\n Error: ` + JSON.stringify(error));
- }
- }
- _hasMasterKey(request, validKeyPairs) {
- if (!validKeyPairs || validKeyPairs.size == 0 || !validKeyPairs.has('masterKey')) {
- return false;
- }
- if (!request || !Object.prototype.hasOwnProperty.call(request, 'masterKey')) {
- return false;
- }
- return request.masterKey === validKeyPairs.get('masterKey');
- }
- _validateKeys(request, validKeyPairs) {
- if (!validKeyPairs || validKeyPairs.size == 0) {
- return true;
- }
- let isValid = false;
- for (const [key, secret] of validKeyPairs) {
- if (!request[key] || request[key] !== secret) {
- continue;
- }
- isValid = true;
- break;
- }
- return isValid;
- }
- async _handleSubscribe(parseWebsocket, request) {
- // If we can not find this client, return error to client
- if (!Object.prototype.hasOwnProperty.call(parseWebsocket, 'clientId')) {
- _Client.Client.pushError(parseWebsocket, 2, 'Can not find this client, make sure you connect to server before subscribing');
- _logger.default.error('Can not find this client, make sure you connect to server before subscribing');
- return;
- }
- const client = this.clients.get(parseWebsocket.clientId);
- const className = request.query.className;
- let authCalled = false;
- try {
- const trigger = (0, _triggers.getTrigger)(className, 'beforeSubscribe', _node.default.applicationId);
- if (trigger) {
- const auth = await this.getAuthFromClient(client, request.requestId, request.sessionToken);
- authCalled = true;
- if (auth && auth.user) {
- request.user = auth.user;
- }
- const parseQuery = new _node.default.Query(className);
- parseQuery.withJSON(request.query);
- request.query = parseQuery;
- await (0, _triggers.runTrigger)(trigger, `beforeSubscribe.${className}`, request, auth);
- const query = request.query.toJSON();
- request.query = query;
- }
- if (className === '_Session') {
- if (!authCalled) {
- const auth = await this.getAuthFromClient(client, request.requestId, request.sessionToken);
- if (auth && auth.user) {
- request.user = auth.user;
- }
- }
- if (request.user) {
- request.query.where.user = request.user.toPointer();
- } else if (!request.master) {
- _Client.Client.pushError(parseWebsocket, _node.default.Error.INVALID_SESSION_TOKEN, 'Invalid session token', false, request.requestId);
- return;
- }
- }
- // Get subscription from subscriptions, create one if necessary
- const subscriptionHash = (0, _QueryTools.queryHash)(request.query);
- // Add className to subscriptions if necessary
- if (!this.subscriptions.has(className)) {
- this.subscriptions.set(className, new Map());
- }
- const classSubscriptions = this.subscriptions.get(className);
- let subscription;
- if (classSubscriptions.has(subscriptionHash)) {
- subscription = classSubscriptions.get(subscriptionHash);
- } else {
- subscription = new _Subscription.Subscription(className, request.query.where, subscriptionHash);
- classSubscriptions.set(subscriptionHash, subscription);
- }
- // Add subscriptionInfo to client
- const subscriptionInfo = {
- subscription: subscription
- };
- // Add selected fields, sessionToken and installationId for this subscription if necessary
- if (request.query.keys) {
- subscriptionInfo.keys = Array.isArray(request.query.keys) ? request.query.keys : request.query.keys.split(',');
- }
- if (request.query.watch) {
- subscriptionInfo.watch = request.query.watch;
- }
- if (request.sessionToken) {
- subscriptionInfo.sessionToken = request.sessionToken;
- }
- client.addSubscriptionInfo(request.requestId, subscriptionInfo);
- // Add clientId to subscription
- subscription.addClientSubscription(parseWebsocket.clientId, request.requestId);
- client.pushSubscribe(request.requestId);
- _logger.default.verbose(`Create client ${parseWebsocket.clientId} new subscription: ${request.requestId}`);
- _logger.default.verbose('Current client number: %d', this.clients.size);
- (0, _triggers.runLiveQueryEventHandlers)({
- client,
- event: 'subscribe',
- clients: this.clients.size,
- subscriptions: this.subscriptions.size,
- sessionToken: request.sessionToken,
- useMasterKey: client.hasMasterKey,
- installationId: client.installationId
- });
- } catch (e) {
- const error = (0, _triggers.resolveError)(e);
- _Client.Client.pushError(parseWebsocket, error.code, error.message, false, request.requestId);
- _logger.default.error(`Failed running beforeSubscribe on ${className} for session ${request.sessionToken} with:\n Error: ` + JSON.stringify(error));
- }
- }
- _handleUpdateSubscription(parseWebsocket, request) {
- this._handleUnsubscribe(parseWebsocket, request, false);
- this._handleSubscribe(parseWebsocket, request);
- }
- _handleUnsubscribe(parseWebsocket, request, notifyClient = true) {
- // If we can not find this client, return error to client
- if (!Object.prototype.hasOwnProperty.call(parseWebsocket, 'clientId')) {
- _Client.Client.pushError(parseWebsocket, 2, 'Can not find this client, make sure you connect to server before unsubscribing');
- _logger.default.error('Can not find this client, make sure you connect to server before unsubscribing');
- return;
- }
- const requestId = request.requestId;
- const client = this.clients.get(parseWebsocket.clientId);
- if (typeof client === 'undefined') {
- _Client.Client.pushError(parseWebsocket, 2, 'Cannot find client with clientId ' + parseWebsocket.clientId + '. Make sure you connect to live query server before unsubscribing.');
- _logger.default.error('Can not find this client ' + parseWebsocket.clientId);
- return;
- }
- const subscriptionInfo = client.getSubscriptionInfo(requestId);
- if (typeof subscriptionInfo === 'undefined') {
- _Client.Client.pushError(parseWebsocket, 2, 'Cannot find subscription with clientId ' + parseWebsocket.clientId + ' subscriptionId ' + requestId + '. Make sure you subscribe to live query server before unsubscribing.');
- _logger.default.error('Can not find subscription with clientId ' + parseWebsocket.clientId + ' subscriptionId ' + requestId);
- return;
- }
- // Remove subscription from client
- client.deleteSubscriptionInfo(requestId);
- // Remove client from subscription
- const subscription = subscriptionInfo.subscription;
- const className = subscription.className;
- subscription.deleteClientSubscription(parseWebsocket.clientId, requestId);
- // If there is no client which is subscribing this subscription, remove it from subscriptions
- const classSubscriptions = this.subscriptions.get(className);
- if (!subscription.hasSubscribingClient()) {
- classSubscriptions.delete(subscription.hash);
- }
- // If there is no subscriptions under this class, remove it from subscriptions
- if (classSubscriptions.size === 0) {
- this.subscriptions.delete(className);
- }
- (0, _triggers.runLiveQueryEventHandlers)({
- client,
- event: 'unsubscribe',
- clients: this.clients.size,
- subscriptions: this.subscriptions.size,
- sessionToken: subscriptionInfo.sessionToken,
- useMasterKey: client.hasMasterKey,
- installationId: client.installationId
- });
- if (!notifyClient) {
- return;
- }
- client.pushUnsubscribe(request.requestId);
- _logger.default.verbose(`Delete client: ${parseWebsocket.clientId} | subscription: ${request.requestId}`);
- }
- }
- exports.ParseLiveQueryServer = ParseLiveQueryServer;
- //# 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":[]}
|