"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":[]}