parse-server 4.10.4 → 5.0.0-alpha.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +451 -153
- package/lib/AccountLockout.js +23 -2
- package/lib/Adapters/AdapterLoader.js +1 -1
- package/lib/Adapters/Analytics/AnalyticsAdapter.js +1 -1
- package/lib/Adapters/Auth/AuthAdapter.js +1 -1
- package/lib/Adapters/Auth/OAuth1Client.js +1 -1
- package/lib/Adapters/Auth/facebook.js +110 -10
- package/lib/Adapters/Auth/gcenter.js +1 -1
- package/lib/Adapters/Auth/gpgames.js +1 -1
- package/lib/Adapters/Auth/instagram.js +4 -2
- package/lib/Adapters/Auth/keycloak.js +1 -1
- package/lib/Adapters/Auth/ldap.js +3 -1
- package/lib/Adapters/Auth/oauth2.js +1 -1
- package/lib/Adapters/Auth/phantauth.js +1 -1
- package/lib/Adapters/Cache/CacheAdapter.js +1 -1
- package/lib/Adapters/Cache/RedisCacheAdapter.js +143 -0
- package/lib/Adapters/Cache/SchemaCache.js +31 -0
- package/lib/Adapters/Email/MailAdapter.js +1 -1
- package/lib/Adapters/Files/FilesAdapter.js +1 -1
- package/lib/Adapters/Files/GridFSBucketAdapter.js +1 -1
- package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
- package/lib/Adapters/Logger/LoggerAdapter.js +1 -1
- package/lib/Adapters/Logger/WinstonLogger.js +4 -4
- package/lib/Adapters/PubSub/EventEmitterPubSub.js +5 -1
- package/lib/Adapters/PubSub/PubSubAdapter.js +1 -1
- package/lib/Adapters/Push/PushAdapter.js +1 -1
- package/lib/Adapters/Storage/Mongo/MongoCollection.js +1 -1
- package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +1 -1
- package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +38 -11
- package/lib/Adapters/Storage/Mongo/MongoTransform.js +9 -6
- package/lib/Adapters/Storage/Postgres/PostgresClient.js +11 -1
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +95 -61
- package/lib/Adapters/WebSocketServer/WSAdapter.js +1 -1
- package/lib/Adapters/WebSocketServer/WSSAdapter.js +1 -1
- package/lib/Auth.js +2 -39
- package/lib/Config.js +149 -8
- package/lib/Controllers/AdaptableController.js +1 -9
- package/lib/Controllers/CacheController.js +1 -1
- package/lib/Controllers/DatabaseController.js +165 -44
- package/lib/Controllers/FilesController.js +1 -1
- package/lib/Controllers/HooksController.js +2 -2
- package/lib/Controllers/LiveQueryController.js +16 -3
- package/lib/Controllers/LoggerController.js +1 -1
- package/lib/Controllers/ParseGraphQLController.js +2 -2
- package/lib/Controllers/PushController.js +1 -1
- package/lib/Controllers/SchemaController.js +101 -88
- package/lib/Controllers/UserController.js +16 -5
- package/lib/Controllers/index.js +10 -11
- package/lib/Deprecator/Deprecations.js +28 -0
- package/lib/Deprecator/Deprecator.js +135 -0
- package/lib/GraphQL/ParseGraphQLSchema.js +71 -39
- package/lib/GraphQL/ParseGraphQLServer.js +3 -3
- package/lib/GraphQL/loaders/defaultGraphQLMutations.js +2 -2
- package/lib/GraphQL/loaders/defaultGraphQLQueries.js +2 -2
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +4 -7
- package/lib/GraphQL/loaders/defaultRelaySchema.js +3 -3
- package/lib/GraphQL/loaders/filesMutations.js +2 -2
- package/lib/GraphQL/loaders/functionsMutations.js +9 -5
- package/lib/GraphQL/loaders/parseClassMutations.js +21 -9
- package/lib/GraphQL/loaders/parseClassQueries.js +9 -6
- package/lib/GraphQL/loaders/parseClassTypes.js +5 -5
- package/lib/GraphQL/loaders/schemaDirectives.js +1 -1
- package/lib/GraphQL/loaders/schemaMutations.js +8 -6
- package/lib/GraphQL/loaders/schemaQueries.js +6 -4
- package/lib/GraphQL/loaders/usersMutations.js +65 -7
- package/lib/GraphQL/transformers/constraintType.js +2 -2
- package/lib/GraphQL/transformers/inputType.js +2 -2
- package/lib/GraphQL/transformers/mutation.js +45 -10
- package/lib/GraphQL/transformers/outputType.js +2 -2
- package/lib/GraphQL/transformers/query.js +2 -2
- package/lib/GraphQL/transformers/schemaFields.js +1 -1
- package/lib/KeyPromiseQueue.js +59 -0
- package/lib/LiveQuery/Client.js +1 -1
- package/lib/LiveQuery/Id.js +1 -1
- package/lib/LiveQuery/ParseLiveQueryServer.js +144 -38
- package/lib/LiveQuery/ParseWebSocketServer.js +2 -2
- package/lib/LiveQuery/QueryTools.js +50 -1
- package/lib/LiveQuery/equalObjects.js +1 -1
- package/lib/Options/Definitions.js +220 -33
- package/lib/Options/docs.js +79 -21
- package/lib/Options/index.js +3 -1
- package/lib/Page.js +53 -0
- package/lib/ParseServer.js +37 -16
- package/lib/ParseServerRESTController.js +55 -45
- package/lib/PromiseRouter.js +7 -20
- package/lib/Push/PushQueue.js +1 -1
- package/lib/Push/PushWorker.js +2 -2
- package/lib/Push/utils.js +1 -1
- package/lib/RestQuery.js +44 -8
- package/lib/RestWrite.js +60 -10
- package/lib/Routers/AggregateRouter.js +23 -18
- package/lib/Routers/AudiencesRouter.js +2 -2
- package/lib/Routers/ClassesRouter.js +11 -11
- package/lib/Routers/CloudCodeRouter.js +1 -1
- package/lib/Routers/FeaturesRouter.js +2 -2
- package/lib/Routers/FilesRouter.js +34 -7
- package/lib/Routers/FunctionsRouter.js +2 -2
- package/lib/Routers/GlobalConfigRouter.js +2 -2
- package/lib/Routers/GraphQLRouter.js +3 -3
- package/lib/Routers/HooksRouter.js +2 -2
- package/lib/Routers/LogsRouter.js +2 -2
- package/lib/Routers/PagesRouter.js +722 -0
- package/lib/Routers/PurgeRouter.js +2 -2
- package/lib/Routers/PushRouter.js +3 -3
- package/lib/Routers/SchemasRouter.js +2 -2
- package/lib/Routers/SecurityRouter.js +47 -0
- package/lib/Routers/SessionsRouter.js +4 -2
- package/lib/Routers/UsersRouter.js +88 -17
- package/lib/Security/Check.js +118 -0
- package/lib/Security/CheckGroup.js +54 -0
- package/lib/Security/CheckGroups/CheckGroupDatabase.js +57 -0
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +82 -0
- package/lib/Security/CheckGroups/CheckGroups.js +24 -0
- package/lib/Security/CheckRunner.js +236 -0
- package/lib/StatusHandler.js +27 -36
- package/lib/TestUtils.js +1 -1
- package/lib/Utils.js +226 -0
- package/lib/batch.js +55 -44
- package/lib/cli/utils/commander.js +8 -3
- package/lib/cloud-code/HTTPResponse.js +1 -1
- package/lib/cloud-code/Parse.Cloud.js +155 -19
- package/lib/cloud-code/httpRequest.js +1 -1
- package/lib/index.js +6 -12
- package/lib/middlewares.js +39 -4
- package/lib/rest.js +4 -4
- package/lib/triggers.js +134 -121
- package/lib/vendor/mongodbUrl.js +8 -10
- package/package.json +54 -29
- package/CHANGELOG.md +0 -1780
- package/lib/Adapters/Cache/RedisCacheAdapter/KeyPromiseQueue.js +0 -59
- package/lib/Adapters/Cache/RedisCacheAdapter/index.js +0 -130
- package/lib/Controllers/SchemaCache.js +0 -75
|
@@ -90,7 +90,7 @@ class ParseLiveQueryServer {
|
|
|
90
90
|
// to the subscribers and the handler will be called.
|
|
91
91
|
|
|
92
92
|
this.subscriber.on('message', (channel, messageStr) => {
|
|
93
|
-
_logger.default.verbose('Subscribe
|
|
93
|
+
_logger.default.verbose('Subscribe message %j', messageStr);
|
|
94
94
|
|
|
95
95
|
let message;
|
|
96
96
|
|
|
@@ -145,7 +145,7 @@ class ParseLiveQueryServer {
|
|
|
145
145
|
// Message.originalParseObject is the original ParseObject.
|
|
146
146
|
|
|
147
147
|
|
|
148
|
-
_onAfterDelete(message) {
|
|
148
|
+
async _onAfterDelete(message) {
|
|
149
149
|
_logger.default.verbose(_node.default.applicationId + 'afterDelete is triggered');
|
|
150
150
|
|
|
151
151
|
let deletedParseObject = message.currentParseObject.toJSON();
|
|
@@ -178,17 +178,17 @@ class ParseLiveQueryServer {
|
|
|
178
178
|
continue;
|
|
179
179
|
}
|
|
180
180
|
|
|
181
|
-
|
|
181
|
+
requestIds.forEach(async requestId => {
|
|
182
182
|
const acl = message.currentParseObject.getACL(); // Check CLP
|
|
183
183
|
|
|
184
184
|
const op = this._getCLPOperation(subscription.query);
|
|
185
185
|
|
|
186
186
|
let res = {};
|
|
187
187
|
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
188
|
+
try {
|
|
189
|
+
await this._matchesCLP(classLevelPermissions, message.currentParseObject, client, requestId, op);
|
|
190
|
+
const isMatched = await this._matchesACL(acl, client, requestId);
|
|
191
|
+
|
|
192
192
|
if (!isMatched) {
|
|
193
193
|
return null;
|
|
194
194
|
}
|
|
@@ -203,15 +203,28 @@ class ParseLiveQueryServer {
|
|
|
203
203
|
installationId: client.installationId,
|
|
204
204
|
sendEvent: true
|
|
205
205
|
};
|
|
206
|
-
|
|
207
|
-
|
|
206
|
+
const trigger = (0, _triggers.getTrigger)(className, 'afterEvent', _node.default.applicationId);
|
|
207
|
+
|
|
208
|
+
if (trigger) {
|
|
209
|
+
const auth = await this.getAuthFromClient(client, requestId);
|
|
210
|
+
|
|
211
|
+
if (auth && auth.user) {
|
|
212
|
+
res.user = auth.user;
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
if (res.object) {
|
|
216
|
+
res.object = _node.default.Object.fromJSON(res.object);
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
await (0, _triggers.runTrigger)(trigger, `afterEvent.${className}`, res, auth);
|
|
220
|
+
}
|
|
221
|
+
|
|
208
222
|
if (!res.sendEvent) {
|
|
209
223
|
return;
|
|
210
224
|
}
|
|
211
225
|
|
|
212
226
|
if (res.object && typeof res.object.toJSON === 'function') {
|
|
213
|
-
deletedParseObject = res.object.
|
|
214
|
-
deletedParseObject.className = className;
|
|
227
|
+
deletedParseObject = (0, _triggers.toJSONwithObjects)(res.object, res.object.className || className);
|
|
215
228
|
}
|
|
216
229
|
|
|
217
230
|
if ((deletedParseObject.className === '_User' || deletedParseObject.className === '_Session') && !client.hasMasterKey) {
|
|
@@ -220,19 +233,19 @@ class ParseLiveQueryServer {
|
|
|
220
233
|
}
|
|
221
234
|
|
|
222
235
|
client.pushDelete(requestId, deletedParseObject);
|
|
223
|
-
}
|
|
224
|
-
_Client.Client.pushError(client.parseWebSocket, error.code ||
|
|
236
|
+
} catch (error) {
|
|
237
|
+
_Client.Client.pushError(client.parseWebSocket, error.code || _node.default.Error.SCRIPT_FAILED, error.message || error, false, requestId);
|
|
225
238
|
|
|
226
239
|
_logger.default.error(`Failed running afterLiveQueryEvent on class ${className} for event ${res.event} with session ${res.sessionToken} with:\n Error: ` + JSON.stringify(error));
|
|
227
|
-
}
|
|
228
|
-
}
|
|
240
|
+
}
|
|
241
|
+
});
|
|
229
242
|
}
|
|
230
243
|
}
|
|
231
244
|
} // Message is the JSON object from publisher after inflated. Message.currentParseObject is the ParseObject after changes.
|
|
232
245
|
// Message.originalParseObject is the original ParseObject.
|
|
233
246
|
|
|
234
247
|
|
|
235
|
-
_onAfterSave(message) {
|
|
248
|
+
async _onAfterSave(message) {
|
|
236
249
|
_logger.default.verbose(_node.default.applicationId + 'afterSave is triggered');
|
|
237
250
|
|
|
238
251
|
let originalParseObject = null;
|
|
@@ -269,7 +282,7 @@ class ParseLiveQueryServer {
|
|
|
269
282
|
continue;
|
|
270
283
|
}
|
|
271
284
|
|
|
272
|
-
|
|
285
|
+
requestIds.forEach(async requestId => {
|
|
273
286
|
// Set orignal ParseObject ACL checking promise, if the object does not match
|
|
274
287
|
// subscription, we do not need to check ACL
|
|
275
288
|
let originalACLCheckingPromise;
|
|
@@ -298,11 +311,12 @@ class ParseLiveQueryServer {
|
|
|
298
311
|
currentACLCheckingPromise = this._matchesACL(currentACL, client, requestId);
|
|
299
312
|
}
|
|
300
313
|
|
|
301
|
-
|
|
314
|
+
try {
|
|
315
|
+
const op = this._getCLPOperation(subscription.query);
|
|
316
|
+
|
|
317
|
+
await this._matchesCLP(classLevelPermissions, message.currentParseObject, client, requestId, op);
|
|
318
|
+
const [isOriginalMatched, isCurrentMatched] = await Promise.all([originalACLCheckingPromise, currentACLCheckingPromise]);
|
|
302
319
|
|
|
303
|
-
this._matchesCLP(classLevelPermissions, message.currentParseObject, client, requestId, op).then(() => {
|
|
304
|
-
return Promise.all([originalACLCheckingPromise, currentACLCheckingPromise]);
|
|
305
|
-
}).then(([isOriginalMatched, isCurrentMatched]) => {
|
|
306
320
|
_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
|
|
307
321
|
|
|
308
322
|
|
|
@@ -322,7 +336,6 @@ class ParseLiveQueryServer {
|
|
|
322
336
|
return null;
|
|
323
337
|
}
|
|
324
338
|
|
|
325
|
-
message.event = type;
|
|
326
339
|
res = {
|
|
327
340
|
event: type,
|
|
328
341
|
sessionToken: client.sessionToken,
|
|
@@ -334,20 +347,36 @@ class ParseLiveQueryServer {
|
|
|
334
347
|
installationId: client.installationId,
|
|
335
348
|
sendEvent: true
|
|
336
349
|
};
|
|
337
|
-
|
|
338
|
-
|
|
350
|
+
const trigger = (0, _triggers.getTrigger)(className, 'afterEvent', _node.default.applicationId);
|
|
351
|
+
|
|
352
|
+
if (trigger) {
|
|
353
|
+
if (res.object) {
|
|
354
|
+
res.object = _node.default.Object.fromJSON(res.object);
|
|
355
|
+
}
|
|
356
|
+
|
|
357
|
+
if (res.original) {
|
|
358
|
+
res.original = _node.default.Object.fromJSON(res.original);
|
|
359
|
+
}
|
|
360
|
+
|
|
361
|
+
const auth = await this.getAuthFromClient(client, requestId);
|
|
362
|
+
|
|
363
|
+
if (auth && auth.user) {
|
|
364
|
+
res.user = auth.user;
|
|
365
|
+
}
|
|
366
|
+
|
|
367
|
+
await (0, _triggers.runTrigger)(trigger, `afterEvent.${className}`, res, auth);
|
|
368
|
+
}
|
|
369
|
+
|
|
339
370
|
if (!res.sendEvent) {
|
|
340
371
|
return;
|
|
341
372
|
}
|
|
342
373
|
|
|
343
374
|
if (res.object && typeof res.object.toJSON === 'function') {
|
|
344
|
-
currentParseObject = res.object.
|
|
345
|
-
currentParseObject.className = res.object.className || className;
|
|
375
|
+
currentParseObject = (0, _triggers.toJSONwithObjects)(res.object, res.object.className || className);
|
|
346
376
|
}
|
|
347
377
|
|
|
348
378
|
if (res.original && typeof res.original.toJSON === 'function') {
|
|
349
|
-
originalParseObject = res.original.
|
|
350
|
-
originalParseObject.className = res.original.className || className;
|
|
379
|
+
originalParseObject = (0, _triggers.toJSONwithObjects)(res.original, res.original.className || className);
|
|
351
380
|
}
|
|
352
381
|
|
|
353
382
|
if ((currentParseObject.className === '_User' || currentParseObject.className === '_Session') && !client.hasMasterKey) {
|
|
@@ -359,17 +388,17 @@ class ParseLiveQueryServer {
|
|
|
359
388
|
(_originalParseObject2 = originalParseObject) === null || _originalParseObject2 === void 0 ? true : delete _originalParseObject2.authData;
|
|
360
389
|
}
|
|
361
390
|
|
|
362
|
-
const functionName = 'push' +
|
|
391
|
+
const functionName = 'push' + res.event.charAt(0).toUpperCase() + res.event.slice(1);
|
|
363
392
|
|
|
364
393
|
if (client[functionName]) {
|
|
365
394
|
client[functionName](requestId, currentParseObject, originalParseObject);
|
|
366
395
|
}
|
|
367
|
-
}
|
|
368
|
-
_Client.Client.pushError(client.parseWebSocket, error.code ||
|
|
396
|
+
} catch (error) {
|
|
397
|
+
_Client.Client.pushError(client.parseWebSocket, error.code || _node.default.Error.SCRIPT_FAILED, error.message || error, false, requestId);
|
|
369
398
|
|
|
370
399
|
_logger.default.error(`Failed running afterLiveQueryEvent on class ${className} for event ${res.event} with session ${res.sessionToken} with:\n Error: ` + JSON.stringify(error));
|
|
371
|
-
}
|
|
372
|
-
}
|
|
400
|
+
}
|
|
401
|
+
});
|
|
373
402
|
}
|
|
374
403
|
}
|
|
375
404
|
}
|
|
@@ -615,6 +644,31 @@ class ParseLiveQueryServer {
|
|
|
615
644
|
});
|
|
616
645
|
}
|
|
617
646
|
|
|
647
|
+
async getAuthFromClient(client, requestId, sessionToken) {
|
|
648
|
+
const getSessionFromClient = () => {
|
|
649
|
+
const subscriptionInfo = client.getSubscriptionInfo(requestId);
|
|
650
|
+
|
|
651
|
+
if (typeof subscriptionInfo === 'undefined') {
|
|
652
|
+
return client.sessionToken;
|
|
653
|
+
}
|
|
654
|
+
|
|
655
|
+
return subscriptionInfo.sessionToken || client.sessionToken;
|
|
656
|
+
};
|
|
657
|
+
|
|
658
|
+
if (!sessionToken) {
|
|
659
|
+
sessionToken = getSessionFromClient();
|
|
660
|
+
}
|
|
661
|
+
|
|
662
|
+
if (!sessionToken) {
|
|
663
|
+
return;
|
|
664
|
+
}
|
|
665
|
+
|
|
666
|
+
const {
|
|
667
|
+
auth
|
|
668
|
+
} = await this.getAuthForSessionToken(sessionToken);
|
|
669
|
+
return auth;
|
|
670
|
+
}
|
|
671
|
+
|
|
618
672
|
async _matchesACL(acl, client, requestId) {
|
|
619
673
|
// Return true directly if ACL isn't present, ACL is public read, or client has master key
|
|
620
674
|
if (!acl || acl.getPublicReadAccess() || client.hasMasterKey) {
|
|
@@ -666,7 +720,18 @@ class ParseLiveQueryServer {
|
|
|
666
720
|
useMasterKey: client.hasMasterKey,
|
|
667
721
|
installationId: request.installationId
|
|
668
722
|
};
|
|
669
|
-
|
|
723
|
+
const trigger = (0, _triggers.getTrigger)('@Connect', 'beforeConnect', _node.default.applicationId);
|
|
724
|
+
|
|
725
|
+
if (trigger) {
|
|
726
|
+
const auth = await this.getAuthFromClient(client, request.requestId, req.sessionToken);
|
|
727
|
+
|
|
728
|
+
if (auth && auth.user) {
|
|
729
|
+
req.user = auth.user;
|
|
730
|
+
}
|
|
731
|
+
|
|
732
|
+
await (0, _triggers.runTrigger)(trigger, `beforeConnect.@Connect`, req, auth);
|
|
733
|
+
}
|
|
734
|
+
|
|
670
735
|
parseWebsocket.clientId = clientId;
|
|
671
736
|
this.clients.set(parseWebsocket.clientId, client);
|
|
672
737
|
|
|
@@ -675,7 +740,7 @@ class ParseLiveQueryServer {
|
|
|
675
740
|
client.pushConnect();
|
|
676
741
|
(0, _triggers.runLiveQueryEventHandlers)(req);
|
|
677
742
|
} catch (error) {
|
|
678
|
-
_Client.Client.pushError(parseWebsocket, error.code ||
|
|
743
|
+
_Client.Client.pushError(parseWebsocket, error.code || _node.default.Error.SCRIPT_FAILED, error.message || error, false);
|
|
679
744
|
|
|
680
745
|
_logger.default.error(`Failed running beforeConnect for session ${request.sessionToken} with:\n Error: ` + JSON.stringify(error));
|
|
681
746
|
}
|
|
@@ -724,9 +789,50 @@ class ParseLiveQueryServer {
|
|
|
724
789
|
|
|
725
790
|
const client = this.clients.get(parseWebsocket.clientId);
|
|
726
791
|
const className = request.query.className;
|
|
792
|
+
let authCalled = false;
|
|
727
793
|
|
|
728
794
|
try {
|
|
729
|
-
|
|
795
|
+
const trigger = (0, _triggers.getTrigger)(className, 'beforeSubscribe', _node.default.applicationId);
|
|
796
|
+
|
|
797
|
+
if (trigger) {
|
|
798
|
+
const auth = await this.getAuthFromClient(client, request.requestId, request.sessionToken);
|
|
799
|
+
authCalled = true;
|
|
800
|
+
|
|
801
|
+
if (auth && auth.user) {
|
|
802
|
+
request.user = auth.user;
|
|
803
|
+
}
|
|
804
|
+
|
|
805
|
+
const parseQuery = new _node.default.Query(className);
|
|
806
|
+
parseQuery.withJSON(request.query);
|
|
807
|
+
request.query = parseQuery;
|
|
808
|
+
await (0, _triggers.runTrigger)(trigger, `beforeSubscribe.${className}`, request, auth);
|
|
809
|
+
const query = request.query.toJSON();
|
|
810
|
+
|
|
811
|
+
if (query.keys) {
|
|
812
|
+
query.fields = query.keys.split(',');
|
|
813
|
+
}
|
|
814
|
+
|
|
815
|
+
request.query = query;
|
|
816
|
+
}
|
|
817
|
+
|
|
818
|
+
if (className === '_Session') {
|
|
819
|
+
if (!authCalled) {
|
|
820
|
+
const auth = await this.getAuthFromClient(client, request.requestId, request.sessionToken);
|
|
821
|
+
|
|
822
|
+
if (auth && auth.user) {
|
|
823
|
+
request.user = auth.user;
|
|
824
|
+
}
|
|
825
|
+
}
|
|
826
|
+
|
|
827
|
+
if (request.user) {
|
|
828
|
+
request.query.where.user = request.user.toPointer();
|
|
829
|
+
} else if (!request.master) {
|
|
830
|
+
_Client.Client.pushError(parseWebsocket, _node.default.Error.INVALID_SESSION_TOKEN, 'Invalid session token', false, request.requestId);
|
|
831
|
+
|
|
832
|
+
return;
|
|
833
|
+
}
|
|
834
|
+
} // Get subscription from subscriptions, create one if necessary
|
|
835
|
+
|
|
730
836
|
|
|
731
837
|
const subscriptionHash = (0, _QueryTools.queryHash)(request.query); // Add className to subscriptions if necessary
|
|
732
838
|
|
|
@@ -776,7 +882,7 @@ class ParseLiveQueryServer {
|
|
|
776
882
|
installationId: client.installationId
|
|
777
883
|
});
|
|
778
884
|
} catch (e) {
|
|
779
|
-
_Client.Client.pushError(parseWebsocket, e.code ||
|
|
885
|
+
_Client.Client.pushError(parseWebsocket, e.code || _node.default.Error.SCRIPT_FAILED, e.message || e, false, request.requestId);
|
|
780
886
|
|
|
781
887
|
_logger.default.error(`Failed running beforeSubscribe on ${className} for session ${request.sessionToken} with:\n Error: ` + JSON.stringify(e));
|
|
782
888
|
}
|
|
@@ -859,4 +965,4 @@ class ParseLiveQueryServer {
|
|
|
859
965
|
}
|
|
860
966
|
|
|
861
967
|
exports.ParseLiveQueryServer = ParseLiveQueryServer;
|
|
862
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/LiveQuery/ParseLiveQueryServer.js"],"names":["ParseLiveQueryServer","constructor","server","config","parseServerConfig","clients","Map","subscriptions","appId","Parse","applicationId","masterKey","keyPairs","key","Object","keys","set","logger","verbose","disableSingleInstance","serverURL","initialize","javaScriptKey","cacheController","cacheTimeout","authCache","LRU","max","maxAge","parseWebSocketServer","ParseWebSocketServer","parseWebsocket","_onConnect","subscriber","ParsePubSub","createSubscriber","subscribe","on","channel","messageStr","message","JSON","parse","e","error","_inflateParseObject","_onAfterSave","_onAfterDelete","currentParseObject","UserRouter","removeHiddenProperties","className","parseObject","_finishFetch","originalParseObject","deletedParseObject","toJSON","classLevelPermissions","id","size","classSubscriptions","get","debug","subscription","values","isSubscriptionMatched","_matchesSubscription","clientId","requestIds","_","entries","clientRequestIds","client","requestId","acl","getACL","op","_getCLPOperation","query","res","_matchesCLP","then","_matchesACL","isMatched","event","sessionToken","object","useMasterKey","hasMasterKey","installationId","sendEvent","authData","pushDelete","catch","Client","pushError","parseWebSocket","code","stringify","isOriginalSubscriptionMatched","isCurrentSubscriptionMatched","originalACLCheckingPromise","Promise","resolve","originalACL","currentACLCheckingPromise","currentACL","all","isOriginalMatched","isCurrentMatched","hash","type","original","functionName","charAt","toUpperCase","slice","request","tv4","validate","RequestSchema","_handleConnect","_handleSubscribe","_handleUpdateSubscription","_handleUnsubscribe","info","has","delete","subscriptionInfo","subscriptionInfos","deleteClientSubscription","hasSubscribingClient","getAuthForSessionToken","fromCache","authPromise","auth","userId","user","result","Error","INVALID_SESSION_TOKEN","del","getSubscriptionInfo","aclGroup","push","SchemaController","validatePermission","length","objectId","_verifyACL","token","isSubscriptionSessionTokenMatched","getReadAccess","acl_has_roles","permissionsById","some","startsWith","roleNames","getUserRoles","role","getPublicReadAccess","subscriptionToken","clientSessionToken","_validateKeys","_hasMasterKey","req","pushConnect","validKeyPairs","prototype","hasOwnProperty","call","isValid","secret","subscriptionHash","Subscription","where","fields","addSubscriptionInfo","addClientSubscription","pushSubscribe","notifyClient","deleteSubscriptionInfo","pushUnsubscribe"],"mappings":"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","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  maybeRunConnectTrigger,\n  maybeRunSubscribeTrigger,\n  maybeRunAfterEventTrigger,\n} from '../triggers';\nimport { getAuthForSessionToken, Auth } from '../Auth';\nimport { getCacheController } from '../Controllers';\nimport LRU from 'lru-cache';\nimport UserRouter from '../Routers/UsersRouter';\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      maxAge: config.cacheTimeout,\n    });\n    // Initialize websocket server\n    this.parseWebSocketServer = new ParseWebSocketServer(\n      server,\n      parseWebsocket => this._onConnect(parseWebsocket),\n      config\n    );\n\n    // Initialize subscriber\n    this.subscriber = ParsePubSub.createSubscriber(config);\n    this.subscriber.subscribe(Parse.applicationId + 'afterSave');\n    this.subscriber.subscribe(Parse.applicationId + 'afterDelete');\n    // Register message handler for subscriber. When publisher get messages, it will publish message\n    // to the subscribers and the handler will be called.\n    this.subscriber.on('message', (channel, messageStr) => {\n      logger.verbose('Subscribe messsage %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      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  }\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  _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    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        for (const requestId of requestIds) {\n          const acl = message.currentParseObject.getACL();\n          // Check CLP\n          const op = this._getCLPOperation(subscription.query);\n          let res = {};\n          this._matchesCLP(classLevelPermissions, message.currentParseObject, client, requestId, op)\n            .then(() => {\n              // Check ACL\n              return this._matchesACL(acl, client, requestId);\n            })\n            .then(isMatched => {\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              return maybeRunAfterEventTrigger('afterEvent', className, res);\n            })\n            .then(() => {\n              if (!res.sendEvent) {\n                return;\n              }\n              if (res.object && typeof res.object.toJSON === 'function') {\n                deletedParseObject = res.object.toJSON();\n                deletedParseObject.className = className;\n              }\n              if (\n                (deletedParseObject.className === '_User' ||\n                  deletedParseObject.className === '_Session') &&\n                !client.hasMasterKey\n              ) {\n                delete deletedParseObject.sessionToken;\n                delete deletedParseObject.authData;\n              }\n              client.pushDelete(requestId, deletedParseObject);\n            })\n            .catch(error => {\n              Client.pushError(\n                client.parseWebSocket,\n                error.code || 141,\n                error.message || error,\n                false,\n                requestId\n              );\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  _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        for (const requestId of requestIds) {\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          const op = this._getCLPOperation(subscription.query);\n          this._matchesCLP(classLevelPermissions, message.currentParseObject, client, requestId, op)\n            .then(() => {\n              return Promise.all([originalACLCheckingPromise, currentACLCheckingPromise]);\n            })\n            .then(([isOriginalMatched, isCurrentMatched]) => {\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              message.event = type;\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              return maybeRunAfterEventTrigger('afterEvent', className, res);\n            })\n            .then(\n              () => {\n                if (!res.sendEvent) {\n                  return;\n                }\n                if (res.object && typeof res.object.toJSON === 'function') {\n                  currentParseObject = res.object.toJSON();\n                  currentParseObject.className = res.object.className || className;\n                }\n\n                if (res.original && typeof res.original.toJSON === 'function') {\n                  originalParseObject = res.original.toJSON();\n                  originalParseObject.className = res.original.className || className;\n                }\n                if (\n                  (currentParseObject.className === '_User' ||\n                    currentParseObject.className === '_Session') &&\n                  !client.hasMasterKey\n                ) {\n                  delete currentParseObject.sessionToken;\n                  delete originalParseObject?.sessionToken;\n                  delete currentParseObject.authData;\n                  delete originalParseObject?.authData;\n                }\n                const functionName =\n                  'push' + message.event.charAt(0).toUpperCase() + message.event.slice(1);\n                if (client[functionName]) {\n                  client[functionName](requestId, currentParseObject, originalParseObject);\n                }\n              },\n              error => {\n                Client.pushError(\n                  client.parseWebSocket,\n                  error.code || 141,\n                  error.message || error,\n                  false,\n                  requestId\n                );\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\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(parseObject, subscription.query);\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.del(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  _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\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 _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      await maybeRunConnectTrigger('beforeConnect', req);\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 (error) {\n      Client.pushError(parseWebsocket, error.code || 141, error.message || error, 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    try {\n      await maybeRunSubscribeTrigger('beforeSubscribe', className, request);\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.fields) {\n        subscriptionInfo.fields = request.query.fields;\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      Client.pushError(parseWebsocket, e.code || 141, e.message || e, false, request.requestId);\n      logger.error(\n        `Failed running beforeSubscribe on ${className} for session ${request.sessionToken} with:\\n Error: ` +\n          JSON.stringify(e)\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"]}
|
|
968
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/LiveQuery/ParseLiveQueryServer.js"],"names":["ParseLiveQueryServer","constructor","server","config","parseServerConfig","clients","Map","subscriptions","appId","Parse","applicationId","masterKey","keyPairs","key","Object","keys","set","logger","verbose","disableSingleInstance","serverURL","initialize","javaScriptKey","cacheController","cacheTimeout","authCache","LRU","max","maxAge","parseWebSocketServer","ParseWebSocketServer","parseWebsocket","_onConnect","subscriber","ParsePubSub","createSubscriber","subscribe","on","channel","messageStr","message","JSON","parse","e","error","_inflateParseObject","_onAfterSave","_onAfterDelete","currentParseObject","UserRouter","removeHiddenProperties","className","parseObject","_finishFetch","originalParseObject","deletedParseObject","toJSON","classLevelPermissions","id","size","classSubscriptions","get","debug","subscription","values","isSubscriptionMatched","_matchesSubscription","clientId","requestIds","_","entries","clientRequestIds","client","forEach","requestId","acl","getACL","op","_getCLPOperation","query","res","_matchesCLP","isMatched","_matchesACL","event","sessionToken","object","useMasterKey","hasMasterKey","installationId","sendEvent","trigger","auth","getAuthFromClient","user","fromJSON","authData","pushDelete","Client","pushError","parseWebSocket","code","Error","SCRIPT_FAILED","stringify","isOriginalSubscriptionMatched","isCurrentSubscriptionMatched","originalACLCheckingPromise","Promise","resolve","originalACL","currentACLCheckingPromise","currentACL","isOriginalMatched","isCurrentMatched","all","hash","type","original","functionName","charAt","toUpperCase","slice","request","tv4","validate","RequestSchema","_handleConnect","_handleSubscribe","_handleUpdateSubscription","_handleUnsubscribe","info","has","delete","subscriptionInfo","subscriptionInfos","deleteClientSubscription","hasSubscribingClient","getAuthForSessionToken","fromCache","authPromise","then","userId","catch","result","INVALID_SESSION_TOKEN","del","getSubscriptionInfo","aclGroup","push","SchemaController","validatePermission","length","objectId","_verifyACL","token","isSubscriptionSessionTokenMatched","getReadAccess","acl_has_roles","permissionsById","some","startsWith","roleNames","getUserRoles","role","getSessionFromClient","getPublicReadAccess","subscriptionToken","clientSessionToken","_validateKeys","_hasMasterKey","req","pushConnect","validKeyPairs","prototype","hasOwnProperty","call","isValid","secret","authCalled","parseQuery","Query","withJSON","fields","split","where","toPointer","master","subscriptionHash","Subscription","addSubscriptionInfo","addClientSubscription","pushSubscribe","notifyClient","deleteSubscriptionInfo","pushUnsubscribe"],"mappings":"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","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 { runLiveQueryEventHandlers, getTrigger, runTrigger, toJSONwithObjects } from '../triggers';\nimport { getAuthForSessionToken, Auth } from '../Auth';\nimport { getCacheController } from '../Controllers';\nimport LRU from 'lru-cache';\nimport UserRouter from '../Routers/UsersRouter';\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      maxAge: config.cacheTimeout,\n    });\n    // Initialize websocket server\n    this.parseWebSocketServer = new ParseWebSocketServer(\n      server,\n      parseWebsocket => this._onConnect(parseWebsocket),\n      config\n    );\n\n    // Initialize subscriber\n    this.subscriber = ParsePubSub.createSubscriber(config);\n    this.subscriber.subscribe(Parse.applicationId + 'afterSave');\n    this.subscriber.subscribe(Parse.applicationId + 'afterDelete');\n    // Register message handler for subscriber. When publisher get messages, it will publish message\n    // to the subscribers and the handler will be called.\n    this.subscriber.on('message', (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      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  }\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            if (\n              (deletedParseObject.className === '_User' ||\n                deletedParseObject.className === '_Session') &&\n              !client.hasMasterKey\n            ) {\n              delete deletedParseObject.sessionToken;\n              delete deletedParseObject.authData;\n            }\n            client.pushDelete(requestId, deletedParseObject);\n          } catch (error) {\n            Client.pushError(\n              client.parseWebSocket,\n              error.code || Parse.Error.SCRIPT_FAILED,\n              error.message || error,\n              false,\n              requestId\n            );\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            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            if (\n              (currentParseObject.className === '_User' ||\n                currentParseObject.className === '_Session') &&\n              !client.hasMasterKey\n            ) {\n              delete currentParseObject.sessionToken;\n              delete originalParseObject?.sessionToken;\n              delete currentParseObject.authData;\n              delete originalParseObject?.authData;\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 (error) {\n            Client.pushError(\n              client.parseWebSocket,\n              error.code || Parse.Error.SCRIPT_FAILED,\n              error.message || error,\n              false,\n              requestId\n            );\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(parseObject, subscription.query);\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.del(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  _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\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  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 (error) {\n      Client.pushError(\n        parseWebsocket,\n        error.code || Parse.Error.SCRIPT_FAILED,\n        error.message || error,\n        false\n      );\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        if (query.keys) {\n          query.fields = query.keys.split(',');\n        }\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.fields) {\n        subscriptionInfo.fields = request.query.fields;\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      Client.pushError(\n        parseWebsocket,\n        e.code || Parse.Error.SCRIPT_FAILED,\n        e.message || e,\n        false,\n        request.requestId\n      );\n      logger.error(\n        `Failed running beforeSubscribe on ${className} for session ${request.sessionToken} with:\\n Error: ` +\n          JSON.stringify(e)\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"]}
|