parse-server 2.8.4 → 8.6.2
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/LICENSE +167 -25
- package/NOTICE +10 -0
- package/README.md +929 -278
- package/lib/AccountLockout.js +47 -30
- package/lib/Adapters/AdapterLoader.js +21 -6
- package/lib/Adapters/Analytics/AnalyticsAdapter.js +15 -12
- package/lib/Adapters/Auth/AuthAdapter.js +116 -13
- package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +99 -0
- package/lib/Adapters/Auth/OAuth1Client.js +27 -46
- package/lib/Adapters/Auth/apple.js +123 -0
- package/lib/Adapters/Auth/facebook.js +162 -35
- package/lib/Adapters/Auth/gcenter.js +217 -0
- package/lib/Adapters/Auth/github.js +118 -48
- package/lib/Adapters/Auth/google.js +160 -51
- package/lib/Adapters/Auth/gpgames.js +125 -0
- package/lib/Adapters/Auth/httpsRequest.js +6 -7
- package/lib/Adapters/Auth/index.js +170 -62
- package/lib/Adapters/Auth/instagram.js +114 -40
- package/lib/Adapters/Auth/janraincapture.js +52 -23
- package/lib/Adapters/Auth/janrainengage.js +19 -36
- package/lib/Adapters/Auth/keycloak.js +148 -0
- package/lib/Adapters/Auth/ldap.js +167 -0
- package/lib/Adapters/Auth/line.js +125 -0
- package/lib/Adapters/Auth/linkedin.js +111 -55
- package/lib/Adapters/Auth/meetup.js +24 -34
- package/lib/Adapters/Auth/mfa.js +324 -0
- package/lib/Adapters/Auth/microsoft.js +111 -0
- package/lib/Adapters/Auth/oauth2.js +97 -162
- package/lib/Adapters/Auth/phantauth.js +53 -0
- package/lib/Adapters/Auth/qq.js +108 -49
- package/lib/Adapters/Auth/spotify.js +107 -55
- package/lib/Adapters/Auth/twitter.js +188 -48
- package/lib/Adapters/Auth/utils.js +28 -0
- package/lib/Adapters/Auth/vkontakte.js +26 -39
- package/lib/Adapters/Auth/wechat.js +106 -44
- package/lib/Adapters/Auth/weibo.js +132 -58
- package/lib/Adapters/Cache/CacheAdapter.js +13 -8
- package/lib/Adapters/Cache/InMemoryCache.js +3 -13
- package/lib/Adapters/Cache/InMemoryCacheAdapter.js +5 -13
- package/lib/Adapters/Cache/LRUCache.js +13 -27
- package/lib/Adapters/Cache/NullCacheAdapter.js +3 -8
- package/lib/Adapters/Cache/RedisCacheAdapter.js +85 -76
- package/lib/Adapters/Cache/SchemaCache.js +25 -0
- package/lib/Adapters/Email/MailAdapter.js +10 -8
- package/lib/Adapters/Files/FilesAdapter.js +83 -25
- package/lib/Adapters/Files/GridFSBucketAdapter.js +231 -0
- package/lib/Adapters/Files/GridStoreAdapter.js +4 -91
- package/lib/Adapters/Logger/LoggerAdapter.js +18 -14
- package/lib/Adapters/Logger/WinstonLogger.js +69 -88
- package/lib/Adapters/Logger/WinstonLoggerAdapter.js +7 -16
- package/lib/Adapters/MessageQueue/EventEmitterMQ.js +8 -26
- package/lib/Adapters/PubSub/EventEmitterPubSub.js +12 -25
- package/lib/Adapters/PubSub/PubSubAdapter.js +34 -0
- package/lib/Adapters/PubSub/RedisPubSub.js +42 -19
- package/lib/Adapters/Push/PushAdapter.js +14 -7
- package/lib/Adapters/Storage/Mongo/MongoCollection.js +137 -45
- package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +158 -63
- package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +320 -168
- package/lib/Adapters/Storage/Mongo/MongoTransform.js +279 -306
- package/lib/Adapters/Storage/Postgres/PostgresClient.js +14 -10
- package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +47 -21
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +854 -468
- package/lib/Adapters/Storage/Postgres/sql/index.js +4 -6
- package/lib/Adapters/Storage/StorageAdapter.js +1 -1
- package/lib/Adapters/WebSocketServer/WSAdapter.js +35 -0
- package/lib/Adapters/WebSocketServer/WSSAdapter.js +66 -0
- package/lib/Auth.js +488 -125
- package/lib/ClientSDK.js +2 -6
- package/lib/Config.js +525 -94
- package/lib/Controllers/AdaptableController.js +5 -25
- package/lib/Controllers/AnalyticsController.js +22 -23
- package/lib/Controllers/CacheController.js +10 -31
- package/lib/Controllers/DatabaseController.js +767 -313
- package/lib/Controllers/FilesController.js +49 -54
- package/lib/Controllers/HooksController.js +80 -84
- package/lib/Controllers/LiveQueryController.js +35 -22
- package/lib/Controllers/LoggerController.js +22 -58
- package/lib/Controllers/ParseGraphQLController.js +293 -0
- package/lib/Controllers/PushController.js +58 -49
- package/lib/Controllers/SchemaController.js +916 -422
- package/lib/Controllers/UserController.js +265 -180
- package/lib/Controllers/index.js +90 -125
- package/lib/Controllers/types.js +1 -1
- package/lib/Deprecator/Deprecations.js +30 -0
- package/lib/Deprecator/Deprecator.js +127 -0
- package/lib/Error.js +48 -0
- package/lib/GraphQL/ParseGraphQLSchema.js +375 -0
- package/lib/GraphQL/ParseGraphQLServer.js +214 -0
- package/lib/GraphQL/helpers/objectsMutations.js +30 -0
- package/lib/GraphQL/helpers/objectsQueries.js +246 -0
- package/lib/GraphQL/loaders/configMutations.js +87 -0
- package/lib/GraphQL/loaders/configQueries.js +79 -0
- package/lib/GraphQL/loaders/defaultGraphQLMutations.js +21 -0
- package/lib/GraphQL/loaders/defaultGraphQLQueries.js +23 -0
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +1098 -0
- package/lib/GraphQL/loaders/defaultRelaySchema.js +53 -0
- package/lib/GraphQL/loaders/filesMutations.js +107 -0
- package/lib/GraphQL/loaders/functionsMutations.js +78 -0
- package/lib/GraphQL/loaders/parseClassMutations.js +268 -0
- package/lib/GraphQL/loaders/parseClassQueries.js +127 -0
- package/lib/GraphQL/loaders/parseClassTypes.js +493 -0
- package/lib/GraphQL/loaders/schemaDirectives.js +62 -0
- package/lib/GraphQL/loaders/schemaMutations.js +162 -0
- package/lib/GraphQL/loaders/schemaQueries.js +81 -0
- package/lib/GraphQL/loaders/schemaTypes.js +341 -0
- package/lib/GraphQL/loaders/usersMutations.js +433 -0
- package/lib/GraphQL/loaders/usersQueries.js +90 -0
- package/lib/GraphQL/parseGraphQLUtils.js +63 -0
- package/lib/GraphQL/transformers/className.js +14 -0
- package/lib/GraphQL/transformers/constraintType.js +53 -0
- package/lib/GraphQL/transformers/inputType.js +51 -0
- package/lib/GraphQL/transformers/mutation.js +274 -0
- package/lib/GraphQL/transformers/outputType.js +51 -0
- package/lib/GraphQL/transformers/query.js +237 -0
- package/lib/GraphQL/transformers/schemaFields.js +99 -0
- package/lib/KeyPromiseQueue.js +48 -0
- package/lib/LiveQuery/Client.js +25 -33
- package/lib/LiveQuery/Id.js +2 -5
- package/lib/LiveQuery/ParseCloudCodePublisher.js +26 -23
- package/lib/LiveQuery/ParseLiveQueryServer.js +560 -285
- package/lib/LiveQuery/ParsePubSub.js +7 -16
- package/lib/LiveQuery/ParseWebSocketServer.js +42 -39
- package/lib/LiveQuery/QueryTools.js +76 -15
- package/lib/LiveQuery/RequestSchema.js +111 -97
- package/lib/LiveQuery/SessionTokenCache.js +23 -36
- package/lib/LiveQuery/Subscription.js +8 -17
- package/lib/LiveQuery/equalObjects.js +2 -3
- package/lib/Options/Definitions.js +1355 -382
- package/lib/Options/docs.js +301 -62
- package/lib/Options/index.js +11 -1
- package/lib/Options/parsers.js +14 -10
- package/lib/Page.js +44 -0
- package/lib/ParseMessageQueue.js +6 -13
- package/lib/ParseServer.js +474 -235
- package/lib/ParseServerRESTController.js +102 -40
- package/lib/PromiseRouter.js +39 -50
- package/lib/Push/PushQueue.js +24 -30
- package/lib/Push/PushWorker.js +32 -56
- package/lib/Push/utils.js +22 -35
- package/lib/RestQuery.js +361 -139
- package/lib/RestWrite.js +713 -344
- package/lib/Routers/AggregateRouter.js +97 -71
- package/lib/Routers/AnalyticsRouter.js +8 -14
- package/lib/Routers/AudiencesRouter.js +16 -35
- package/lib/Routers/ClassesRouter.js +86 -72
- package/lib/Routers/CloudCodeRouter.js +28 -37
- package/lib/Routers/FeaturesRouter.js +22 -25
- package/lib/Routers/FilesRouter.js +266 -171
- package/lib/Routers/FunctionsRouter.js +87 -103
- package/lib/Routers/GlobalConfigRouter.js +94 -33
- package/lib/Routers/GraphQLRouter.js +41 -0
- package/lib/Routers/HooksRouter.js +43 -47
- package/lib/Routers/IAPValidationRouter.js +57 -70
- package/lib/Routers/InstallationsRouter.js +17 -25
- package/lib/Routers/LogsRouter.js +10 -25
- package/lib/Routers/PagesRouter.js +647 -0
- package/lib/Routers/PublicAPIRouter.js +104 -112
- package/lib/Routers/PurgeRouter.js +19 -29
- package/lib/Routers/PushRouter.js +14 -28
- package/lib/Routers/RolesRouter.js +7 -14
- package/lib/Routers/SchemasRouter.js +63 -42
- package/lib/Routers/SecurityRouter.js +34 -0
- package/lib/Routers/SessionsRouter.js +25 -38
- package/lib/Routers/UsersRouter.js +463 -190
- package/lib/SchemaMigrations/DefinedSchemas.js +379 -0
- package/lib/SchemaMigrations/Migrations.js +30 -0
- package/lib/Security/Check.js +109 -0
- package/lib/Security/CheckGroup.js +44 -0
- package/lib/Security/CheckGroups/CheckGroupDatabase.js +44 -0
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +96 -0
- package/lib/Security/CheckGroups/CheckGroups.js +21 -0
- package/lib/Security/CheckRunner.js +213 -0
- package/lib/SharedRest.js +29 -0
- package/lib/StatusHandler.js +96 -93
- package/lib/TestUtils.js +70 -14
- package/lib/Utils.js +468 -0
- package/lib/batch.js +74 -40
- package/lib/cache.js +8 -8
- package/lib/cli/definitions/parse-live-query-server.js +4 -3
- package/lib/cli/definitions/parse-server.js +4 -3
- package/lib/cli/parse-live-query-server.js +9 -17
- package/lib/cli/parse-server.js +49 -47
- package/lib/cli/utils/commander.js +20 -29
- package/lib/cli/utils/runner.js +31 -32
- package/lib/cloud-code/Parse.Cloud.js +711 -36
- package/lib/cloud-code/Parse.Server.js +21 -0
- package/lib/cryptoUtils.js +6 -11
- package/lib/defaults.js +21 -15
- package/lib/deprecated.js +1 -1
- package/lib/index.js +78 -67
- package/lib/logger.js +12 -20
- package/lib/middlewares.js +484 -160
- package/lib/password.js +10 -6
- package/lib/request.js +175 -0
- package/lib/requiredParameter.js +4 -3
- package/lib/rest.js +157 -82
- package/lib/triggers.js +627 -185
- package/lib/vendor/README.md +3 -3
- package/lib/vendor/mongodbUrl.js +224 -137
- package/package.json +135 -57
- package/postinstall.js +38 -50
- package/public_html/invalid_verification_link.html +3 -3
- package/types/@types/@parse/fs-files-adapter/index.d.ts +5 -0
- package/types/@types/deepcopy/index.d.ts +5 -0
- package/types/LiveQuery/ParseLiveQueryServer.d.ts +40 -0
- package/types/Options/index.d.ts +301 -0
- package/types/ParseServer.d.ts +65 -0
- package/types/eslint.config.mjs +30 -0
- package/types/index.d.ts +21 -0
- package/types/logger.d.ts +2 -0
- package/types/tests.ts +44 -0
- package/types/tsconfig.json +24 -0
- package/CHANGELOG.md +0 -1246
- package/PATENTS +0 -37
- package/bin/dev +0 -37
- package/lib/.DS_Store +0 -0
- package/lib/Adapters/Auth/common.js +0 -2
- package/lib/Adapters/Auth/facebookaccountkit.js +0 -69
- package/lib/Controllers/SchemaCache.js +0 -97
- package/lib/LiveQuery/.DS_Store +0 -0
- package/lib/cli/utils/parsers.js +0 -77
- package/lib/cloud-code/.DS_Store +0 -0
- package/lib/cloud-code/HTTPResponse.js +0 -57
- package/lib/cloud-code/Untitled-1 +0 -123
- package/lib/cloud-code/httpRequest.js +0 -102
- package/lib/cloud-code/team.html +0 -123
- package/lib/graphql/ParseClass.js +0 -234
- package/lib/graphql/Schema.js +0 -197
- package/lib/graphql/index.js +0 -1
- package/lib/graphql/types/ACL.js +0 -35
- package/lib/graphql/types/Date.js +0 -25
- package/lib/graphql/types/File.js +0 -24
- package/lib/graphql/types/GeoPoint.js +0 -35
- package/lib/graphql/types/JSONObject.js +0 -30
- package/lib/graphql/types/NumberInput.js +0 -43
- package/lib/graphql/types/NumberQuery.js +0 -42
- package/lib/graphql/types/Pointer.js +0 -35
- package/lib/graphql/types/QueryConstraint.js +0 -61
- package/lib/graphql/types/StringQuery.js +0 -39
- package/lib/graphql/types/index.js +0 -110
package/lib/triggers.js
CHANGED
|
@@ -1,51 +1,66 @@
|
|
|
1
|
-
|
|
1
|
+
"use strict";
|
|
2
2
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", {
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
|
-
exports.Types =
|
|
6
|
+
exports.Types = void 0;
|
|
7
|
+
exports._unregisterAll = _unregisterAll;
|
|
8
|
+
exports.addConnectTrigger = addConnectTrigger;
|
|
7
9
|
exports.addFunction = addFunction;
|
|
8
10
|
exports.addJob = addJob;
|
|
9
|
-
exports.addTrigger = addTrigger;
|
|
10
11
|
exports.addLiveQueryEventHandler = addLiveQueryEventHandler;
|
|
11
|
-
exports.
|
|
12
|
-
exports.
|
|
13
|
-
exports._unregisterAll = _unregisterAll;
|
|
14
|
-
exports.getTrigger = getTrigger;
|
|
15
|
-
exports.triggerExists = triggerExists;
|
|
12
|
+
exports.addTrigger = addTrigger;
|
|
13
|
+
exports.getClassName = getClassName;
|
|
16
14
|
exports.getFunction = getFunction;
|
|
15
|
+
exports.getFunctionNames = getFunctionNames;
|
|
17
16
|
exports.getJob = getJob;
|
|
18
17
|
exports.getJobs = getJobs;
|
|
19
|
-
exports.
|
|
18
|
+
exports.getRequestFileObject = getRequestFileObject;
|
|
20
19
|
exports.getRequestObject = getRequestObject;
|
|
21
20
|
exports.getRequestQueryObject = getRequestQueryObject;
|
|
22
21
|
exports.getResponseObject = getResponseObject;
|
|
22
|
+
exports.getTrigger = getTrigger;
|
|
23
|
+
exports.getValidator = getValidator;
|
|
24
|
+
exports.inflate = inflate;
|
|
23
25
|
exports.maybeRunAfterFindTrigger = maybeRunAfterFindTrigger;
|
|
26
|
+
exports.maybeRunFileTrigger = maybeRunFileTrigger;
|
|
27
|
+
exports.maybeRunGlobalConfigTrigger = maybeRunGlobalConfigTrigger;
|
|
24
28
|
exports.maybeRunQueryTrigger = maybeRunQueryTrigger;
|
|
25
29
|
exports.maybeRunTrigger = maybeRunTrigger;
|
|
26
|
-
exports.
|
|
30
|
+
exports.maybeRunValidator = maybeRunValidator;
|
|
31
|
+
exports.removeFunction = removeFunction;
|
|
32
|
+
exports.removeTrigger = removeTrigger;
|
|
33
|
+
exports.resolveError = resolveError;
|
|
27
34
|
exports.runLiveQueryEventHandlers = runLiveQueryEventHandlers;
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
var
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
36
|
-
|
|
35
|
+
exports.runTrigger = runTrigger;
|
|
36
|
+
exports.toJSONwithObjects = toJSONwithObjects;
|
|
37
|
+
exports.triggerExists = triggerExists;
|
|
38
|
+
var _node = _interopRequireDefault(require("parse/node"));
|
|
39
|
+
var _logger = require("./logger");
|
|
40
|
+
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
37
41
|
// triggers.js
|
|
42
|
+
|
|
38
43
|
const Types = exports.Types = {
|
|
44
|
+
beforeLogin: 'beforeLogin',
|
|
45
|
+
afterLogin: 'afterLogin',
|
|
46
|
+
afterLogout: 'afterLogout',
|
|
47
|
+
beforePasswordResetRequest: 'beforePasswordResetRequest',
|
|
39
48
|
beforeSave: 'beforeSave',
|
|
40
49
|
afterSave: 'afterSave',
|
|
41
50
|
beforeDelete: 'beforeDelete',
|
|
42
51
|
afterDelete: 'afterDelete',
|
|
43
52
|
beforeFind: 'beforeFind',
|
|
44
|
-
afterFind: 'afterFind'
|
|
53
|
+
afterFind: 'afterFind',
|
|
54
|
+
beforeConnect: 'beforeConnect',
|
|
55
|
+
beforeSubscribe: 'beforeSubscribe',
|
|
56
|
+
afterEvent: 'afterEvent'
|
|
45
57
|
};
|
|
46
|
-
|
|
58
|
+
const ConnectClassName = '@Connect';
|
|
47
59
|
const baseStore = function () {
|
|
48
|
-
const Validators = {
|
|
60
|
+
const Validators = Object.keys(Types).reduce(function (base, key) {
|
|
61
|
+
base[key] = {};
|
|
62
|
+
return base;
|
|
63
|
+
}, {});
|
|
49
64
|
const Functions = {};
|
|
50
65
|
const Jobs = {};
|
|
51
66
|
const LiveQuery = [];
|
|
@@ -53,7 +68,6 @@ const baseStore = function () {
|
|
|
53
68
|
base[key] = {};
|
|
54
69
|
return base;
|
|
55
70
|
}, {});
|
|
56
|
-
|
|
57
71
|
return Object.freeze({
|
|
58
72
|
Functions,
|
|
59
73
|
Jobs,
|
|
@@ -62,94 +76,180 @@ const baseStore = function () {
|
|
|
62
76
|
LiveQuery
|
|
63
77
|
});
|
|
64
78
|
};
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
79
|
+
function getClassName(parseClass) {
|
|
80
|
+
if (parseClass && parseClass.className) {
|
|
81
|
+
return parseClass.className;
|
|
82
|
+
}
|
|
83
|
+
if (parseClass && parseClass.name) {
|
|
84
|
+
return parseClass.name.replace('Parse', '@');
|
|
70
85
|
}
|
|
86
|
+
return parseClass;
|
|
87
|
+
}
|
|
88
|
+
function validateClassNameForTriggers(className, type) {
|
|
71
89
|
if (type == Types.beforeSave && className === '_PushStatus') {
|
|
72
90
|
// _PushStatus uses undocumented nested key increment ops
|
|
73
91
|
// allowing beforeSave would mess up the objects big time
|
|
74
92
|
// TODO: Allow proper documented way of using nested increment ops
|
|
75
93
|
throw 'Only afterSave is allowed on _PushStatus';
|
|
76
94
|
}
|
|
95
|
+
if ((type === Types.beforeLogin || type === Types.afterLogin || type === Types.beforePasswordResetRequest) && className !== '_User') {
|
|
96
|
+
// TODO: check if upstream code will handle `Error` instance rather
|
|
97
|
+
// than this anti-pattern of throwing strings
|
|
98
|
+
throw 'Only the _User class is allowed for the beforeLogin, afterLogin, and beforePasswordResetRequest triggers';
|
|
99
|
+
}
|
|
100
|
+
if (type === Types.afterLogout && className !== '_Session') {
|
|
101
|
+
// TODO: check if upstream code will handle `Error` instance rather
|
|
102
|
+
// than this anti-pattern of throwing strings
|
|
103
|
+
throw 'Only the _Session class is allowed for the afterLogout trigger.';
|
|
104
|
+
}
|
|
105
|
+
if (className === '_Session' && type !== Types.afterLogout) {
|
|
106
|
+
// TODO: check if upstream code will handle `Error` instance rather
|
|
107
|
+
// than this anti-pattern of throwing strings
|
|
108
|
+
throw 'Only the afterLogout trigger is allowed for the _Session class.';
|
|
109
|
+
}
|
|
77
110
|
return className;
|
|
78
111
|
}
|
|
79
|
-
|
|
80
112
|
const _triggerStore = {};
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
113
|
+
const Category = {
|
|
114
|
+
Functions: 'Functions',
|
|
115
|
+
Validators: 'Validators',
|
|
116
|
+
Jobs: 'Jobs',
|
|
117
|
+
Triggers: 'Triggers'
|
|
118
|
+
};
|
|
119
|
+
function getStore(category, name, applicationId) {
|
|
120
|
+
const invalidNameRegex = /['"`]/;
|
|
121
|
+
if (invalidNameRegex.test(name)) {
|
|
122
|
+
// Prevent a malicious user from injecting properties into the store
|
|
123
|
+
return {};
|
|
124
|
+
}
|
|
125
|
+
const path = name.split('.');
|
|
126
|
+
path.splice(-1); // remove last component
|
|
127
|
+
applicationId = applicationId || _node.default.applicationId;
|
|
84
128
|
_triggerStore[applicationId] = _triggerStore[applicationId] || baseStore();
|
|
85
|
-
_triggerStore[applicationId]
|
|
86
|
-
|
|
129
|
+
let store = _triggerStore[applicationId][category];
|
|
130
|
+
for (const component of path) {
|
|
131
|
+
store = store[component];
|
|
132
|
+
if (!store) {
|
|
133
|
+
return {};
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
return store;
|
|
137
|
+
}
|
|
138
|
+
function add(category, name, handler, applicationId) {
|
|
139
|
+
const lastComponent = name.split('.').splice(-1);
|
|
140
|
+
const store = getStore(category, name, applicationId);
|
|
141
|
+
if (store[lastComponent]) {
|
|
142
|
+
_logger.logger.warn(`Warning: Duplicate cloud functions exist for ${lastComponent}. Only the last one will be used and the others will be ignored.`);
|
|
143
|
+
}
|
|
144
|
+
store[lastComponent] = handler;
|
|
145
|
+
}
|
|
146
|
+
function remove(category, name, applicationId) {
|
|
147
|
+
const lastComponent = name.split('.').splice(-1);
|
|
148
|
+
const store = getStore(category, name, applicationId);
|
|
149
|
+
delete store[lastComponent];
|
|
150
|
+
}
|
|
151
|
+
function get(category, name, applicationId) {
|
|
152
|
+
const lastComponent = name.split('.').splice(-1);
|
|
153
|
+
const store = getStore(category, name, applicationId);
|
|
154
|
+
return store[lastComponent];
|
|
155
|
+
}
|
|
156
|
+
function addFunction(functionName, handler, validationHandler, applicationId) {
|
|
157
|
+
add(Category.Functions, functionName, handler, applicationId);
|
|
158
|
+
add(Category.Validators, functionName, validationHandler, applicationId);
|
|
87
159
|
}
|
|
88
|
-
|
|
89
160
|
function addJob(jobName, handler, applicationId) {
|
|
90
|
-
|
|
91
|
-
_triggerStore[applicationId] = _triggerStore[applicationId] || baseStore();
|
|
92
|
-
_triggerStore[applicationId].Jobs[jobName] = handler;
|
|
161
|
+
add(Category.Jobs, jobName, handler, applicationId);
|
|
93
162
|
}
|
|
94
|
-
|
|
95
|
-
function addTrigger(type, className, handler, applicationId) {
|
|
163
|
+
function addTrigger(type, className, handler, applicationId, validationHandler) {
|
|
96
164
|
validateClassNameForTriggers(className, type);
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
165
|
+
add(Category.Triggers, `${type}.${className}`, handler, applicationId);
|
|
166
|
+
add(Category.Validators, `${type}.${className}`, validationHandler, applicationId);
|
|
167
|
+
}
|
|
168
|
+
function addConnectTrigger(type, handler, applicationId, validationHandler) {
|
|
169
|
+
add(Category.Triggers, `${type}.${ConnectClassName}`, handler, applicationId);
|
|
170
|
+
add(Category.Validators, `${type}.${ConnectClassName}`, validationHandler, applicationId);
|
|
100
171
|
}
|
|
101
|
-
|
|
102
172
|
function addLiveQueryEventHandler(handler, applicationId) {
|
|
103
|
-
applicationId = applicationId ||
|
|
173
|
+
applicationId = applicationId || _node.default.applicationId;
|
|
104
174
|
_triggerStore[applicationId] = _triggerStore[applicationId] || baseStore();
|
|
105
175
|
_triggerStore[applicationId].LiveQuery.push(handler);
|
|
106
176
|
}
|
|
107
|
-
|
|
108
177
|
function removeFunction(functionName, applicationId) {
|
|
109
|
-
|
|
110
|
-
delete _triggerStore[applicationId].Functions[functionName];
|
|
178
|
+
remove(Category.Functions, functionName, applicationId);
|
|
111
179
|
}
|
|
112
|
-
|
|
113
180
|
function removeTrigger(type, className, applicationId) {
|
|
114
|
-
|
|
115
|
-
delete _triggerStore[applicationId].Triggers[type][className];
|
|
181
|
+
remove(Category.Triggers, `${type}.${className}`, applicationId);
|
|
116
182
|
}
|
|
117
|
-
|
|
118
183
|
function _unregisterAll() {
|
|
119
184
|
Object.keys(_triggerStore).forEach(appId => delete _triggerStore[appId]);
|
|
120
185
|
}
|
|
121
|
-
|
|
186
|
+
function toJSONwithObjects(object, className) {
|
|
187
|
+
if (!object || !object.toJSON) {
|
|
188
|
+
return {};
|
|
189
|
+
}
|
|
190
|
+
const toJSON = object.toJSON();
|
|
191
|
+
const stateController = _node.default.CoreManager.getObjectStateController();
|
|
192
|
+
const [pending] = stateController.getPendingOps(object._getStateIdentifier());
|
|
193
|
+
for (const key in pending) {
|
|
194
|
+
const val = object.get(key);
|
|
195
|
+
if (!val || !val._toFullJSON) {
|
|
196
|
+
toJSON[key] = val;
|
|
197
|
+
continue;
|
|
198
|
+
}
|
|
199
|
+
toJSON[key] = val._toFullJSON();
|
|
200
|
+
}
|
|
201
|
+
// Preserve original object's className if no override className is provided
|
|
202
|
+
if (className) {
|
|
203
|
+
toJSON.className = className;
|
|
204
|
+
} else if (object.className && !toJSON.className) {
|
|
205
|
+
toJSON.className = object.className;
|
|
206
|
+
}
|
|
207
|
+
return toJSON;
|
|
208
|
+
}
|
|
122
209
|
function getTrigger(className, triggerType, applicationId) {
|
|
123
210
|
if (!applicationId) {
|
|
124
|
-
throw
|
|
211
|
+
throw 'Missing ApplicationID';
|
|
125
212
|
}
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
213
|
+
return get(Category.Triggers, `${triggerType}.${className}`, applicationId);
|
|
214
|
+
}
|
|
215
|
+
async function runTrigger(trigger, name, request, auth) {
|
|
216
|
+
if (!trigger) {
|
|
217
|
+
return;
|
|
129
218
|
}
|
|
130
|
-
|
|
219
|
+
await maybeRunValidator(request, name, auth);
|
|
220
|
+
if (request.skipWithMasterKey) {
|
|
221
|
+
return;
|
|
222
|
+
}
|
|
223
|
+
return await trigger(request);
|
|
131
224
|
}
|
|
132
|
-
|
|
133
225
|
function triggerExists(className, type, applicationId) {
|
|
134
226
|
return getTrigger(className, type, applicationId) != undefined;
|
|
135
227
|
}
|
|
136
|
-
|
|
137
228
|
function getFunction(functionName, applicationId) {
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
}
|
|
142
|
-
|
|
229
|
+
return get(Category.Functions, functionName, applicationId);
|
|
230
|
+
}
|
|
231
|
+
function getFunctionNames(applicationId) {
|
|
232
|
+
const store = _triggerStore[applicationId] && _triggerStore[applicationId][Category.Functions] || {};
|
|
233
|
+
const functionNames = [];
|
|
234
|
+
const extractFunctionNames = (namespace, store) => {
|
|
235
|
+
Object.keys(store).forEach(name => {
|
|
236
|
+
const value = store[name];
|
|
237
|
+
if (namespace) {
|
|
238
|
+
name = `${namespace}.${name}`;
|
|
239
|
+
}
|
|
240
|
+
if (typeof value === 'function') {
|
|
241
|
+
functionNames.push(name);
|
|
242
|
+
} else {
|
|
243
|
+
extractFunctionNames(name, value);
|
|
244
|
+
}
|
|
245
|
+
});
|
|
246
|
+
};
|
|
247
|
+
extractFunctionNames(null, store);
|
|
248
|
+
return functionNames;
|
|
143
249
|
}
|
|
144
|
-
|
|
145
250
|
function getJob(jobName, applicationId) {
|
|
146
|
-
|
|
147
|
-
if (manager && manager.Jobs) {
|
|
148
|
-
return manager.Jobs[jobName];
|
|
149
|
-
}
|
|
150
|
-
return undefined;
|
|
251
|
+
return get(Category.Jobs, jobName, applicationId);
|
|
151
252
|
}
|
|
152
|
-
|
|
153
253
|
function getJobs(applicationId) {
|
|
154
254
|
var manager = _triggerStore[applicationId];
|
|
155
255
|
if (manager && manager.Jobs) {
|
|
@@ -157,29 +257,29 @@ function getJobs(applicationId) {
|
|
|
157
257
|
}
|
|
158
258
|
return undefined;
|
|
159
259
|
}
|
|
160
|
-
|
|
161
260
|
function getValidator(functionName, applicationId) {
|
|
162
|
-
|
|
163
|
-
if (manager && manager.Validators) {
|
|
164
|
-
return manager.Validators[functionName];
|
|
165
|
-
}
|
|
166
|
-
return undefined;
|
|
261
|
+
return get(Category.Validators, functionName, applicationId);
|
|
167
262
|
}
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
var request = {
|
|
263
|
+
function getRequestObject(triggerType, auth, parseObject, originalParseObject, config, context, isGet) {
|
|
264
|
+
const request = {
|
|
171
265
|
triggerName: triggerType,
|
|
172
266
|
object: parseObject,
|
|
173
267
|
master: false,
|
|
174
268
|
log: config.loggerController,
|
|
175
269
|
headers: config.headers,
|
|
176
|
-
ip: config.ip
|
|
270
|
+
ip: config.ip,
|
|
271
|
+
config
|
|
177
272
|
};
|
|
178
|
-
|
|
273
|
+
if (isGet !== undefined) {
|
|
274
|
+
request.isGet = !!isGet;
|
|
275
|
+
}
|
|
179
276
|
if (originalParseObject) {
|
|
180
277
|
request.original = originalParseObject;
|
|
181
278
|
}
|
|
182
|
-
|
|
279
|
+
if (triggerType === Types.beforeSave || triggerType === Types.afterSave || triggerType === Types.beforeDelete || triggerType === Types.afterDelete || triggerType === Types.beforeLogin || triggerType === Types.afterLogin || triggerType === Types.beforePasswordResetRequest || triggerType === Types.afterFind) {
|
|
280
|
+
// Set a copy of the context on the request object.
|
|
281
|
+
request.context = Object.assign({}, context);
|
|
282
|
+
}
|
|
183
283
|
if (!auth) {
|
|
184
284
|
return request;
|
|
185
285
|
}
|
|
@@ -194,10 +294,8 @@ function getRequestObject(triggerType, auth, parseObject, originalParseObject, c
|
|
|
194
294
|
}
|
|
195
295
|
return request;
|
|
196
296
|
}
|
|
197
|
-
|
|
198
|
-
function getRequestQueryObject(triggerType, auth, query, count, config, isGet) {
|
|
297
|
+
function getRequestQueryObject(triggerType, auth, query, count, config, context, isGet) {
|
|
199
298
|
isGet = !!isGet;
|
|
200
|
-
|
|
201
299
|
var request = {
|
|
202
300
|
triggerName: triggerType,
|
|
203
301
|
query,
|
|
@@ -206,9 +304,10 @@ function getRequestQueryObject(triggerType, auth, query, count, config, isGet) {
|
|
|
206
304
|
log: config.loggerController,
|
|
207
305
|
isGet,
|
|
208
306
|
headers: config.headers,
|
|
209
|
-
ip: config.ip
|
|
307
|
+
ip: config.ip,
|
|
308
|
+
context: context || {},
|
|
309
|
+
config
|
|
210
310
|
};
|
|
211
|
-
|
|
212
311
|
if (!auth) {
|
|
213
312
|
return request;
|
|
214
313
|
}
|
|
@@ -236,102 +335,139 @@ function getResponseObject(request, resolve, reject) {
|
|
|
236
335
|
response = request.objects;
|
|
237
336
|
}
|
|
238
337
|
response = response.map(object => {
|
|
239
|
-
return object
|
|
338
|
+
return toJSONwithObjects(object);
|
|
240
339
|
});
|
|
241
340
|
return resolve(response);
|
|
242
341
|
}
|
|
243
342
|
// Use the JSON response
|
|
244
|
-
if (response && !request.object.equals(response) && request.triggerName === Types.beforeSave) {
|
|
343
|
+
if (response && typeof response === 'object' && !request.object.equals(response) && request.triggerName === Types.beforeSave) {
|
|
344
|
+
return resolve(response);
|
|
345
|
+
}
|
|
346
|
+
if (response && typeof response === 'object' && request.triggerName === Types.afterSave) {
|
|
245
347
|
return resolve(response);
|
|
246
348
|
}
|
|
349
|
+
if (request.triggerName === Types.afterSave) {
|
|
350
|
+
return resolve();
|
|
351
|
+
}
|
|
247
352
|
response = {};
|
|
248
353
|
if (request.triggerName === Types.beforeSave) {
|
|
249
354
|
response['object'] = request.object._getSaveJSON();
|
|
355
|
+
response['object']['objectId'] = request.object.id;
|
|
250
356
|
}
|
|
251
357
|
return resolve(response);
|
|
252
358
|
},
|
|
253
|
-
error: function (
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
code = _node2.default.Error.SCRIPT_FAILED;
|
|
260
|
-
}
|
|
261
|
-
var scriptError = new _node2.default.Error(code, message);
|
|
262
|
-
return reject(scriptError);
|
|
359
|
+
error: function (error) {
|
|
360
|
+
const e = resolveError(error, {
|
|
361
|
+
code: _node.default.Error.SCRIPT_FAILED,
|
|
362
|
+
message: 'Script failed. Unknown error.'
|
|
363
|
+
});
|
|
364
|
+
reject(e);
|
|
263
365
|
}
|
|
264
366
|
};
|
|
265
367
|
}
|
|
266
|
-
|
|
267
368
|
function userIdForLog(auth) {
|
|
268
369
|
return auth && auth.user ? auth.user.id : undefined;
|
|
269
370
|
}
|
|
270
|
-
|
|
271
|
-
|
|
371
|
+
function logTriggerAfterHook(triggerType, className, input, auth, logLevel) {
|
|
372
|
+
if (logLevel === 'silent') {
|
|
373
|
+
return;
|
|
374
|
+
}
|
|
272
375
|
const cleanInput = _logger.logger.truncateLogMessage(JSON.stringify(input));
|
|
273
|
-
_logger.logger
|
|
376
|
+
_logger.logger[logLevel](`${triggerType} triggered for ${className} for user ${userIdForLog(auth)}:\n Input: ${cleanInput}`, {
|
|
274
377
|
className,
|
|
275
378
|
triggerType,
|
|
276
379
|
user: userIdForLog(auth)
|
|
277
380
|
});
|
|
278
381
|
}
|
|
279
|
-
|
|
280
|
-
|
|
382
|
+
function logTriggerSuccessBeforeHook(triggerType, className, input, result, auth, logLevel) {
|
|
383
|
+
if (logLevel === 'silent') {
|
|
384
|
+
return;
|
|
385
|
+
}
|
|
281
386
|
const cleanInput = _logger.logger.truncateLogMessage(JSON.stringify(input));
|
|
282
387
|
const cleanResult = _logger.logger.truncateLogMessage(JSON.stringify(result));
|
|
283
|
-
_logger.logger
|
|
388
|
+
_logger.logger[logLevel](`${triggerType} triggered for ${className} for user ${userIdForLog(auth)}:\n Input: ${cleanInput}\n Result: ${cleanResult}`, {
|
|
284
389
|
className,
|
|
285
390
|
triggerType,
|
|
286
391
|
user: userIdForLog(auth)
|
|
287
392
|
});
|
|
288
393
|
}
|
|
289
|
-
|
|
290
|
-
|
|
394
|
+
function logTriggerErrorBeforeHook(triggerType, className, input, auth, error, logLevel) {
|
|
395
|
+
if (logLevel === 'silent') {
|
|
396
|
+
return;
|
|
397
|
+
}
|
|
291
398
|
const cleanInput = _logger.logger.truncateLogMessage(JSON.stringify(input));
|
|
292
|
-
_logger.logger
|
|
399
|
+
_logger.logger[logLevel](`${triggerType} failed for ${className} for user ${userIdForLog(auth)}:\n Input: ${cleanInput}\n Error: ${JSON.stringify(error)}`, {
|
|
293
400
|
className,
|
|
294
401
|
triggerType,
|
|
295
402
|
error,
|
|
296
403
|
user: userIdForLog(auth)
|
|
297
404
|
});
|
|
298
405
|
}
|
|
299
|
-
|
|
300
|
-
function maybeRunAfterFindTrigger(triggerType, auth, className, objects, config) {
|
|
406
|
+
function maybeRunAfterFindTrigger(triggerType, auth, classNameQuery, objectsInput, config, query, context, isGet) {
|
|
301
407
|
return new Promise((resolve, reject) => {
|
|
302
|
-
const trigger = getTrigger(
|
|
408
|
+
const trigger = getTrigger(classNameQuery, triggerType, config.applicationId);
|
|
303
409
|
if (!trigger) {
|
|
304
|
-
|
|
410
|
+
if (objectsInput && objectsInput.length > 0 && objectsInput[0] instanceof _node.default.Object) {
|
|
411
|
+
return resolve(objectsInput.map(obj => toJSONwithObjects(obj)));
|
|
412
|
+
}
|
|
413
|
+
return resolve(objectsInput || []);
|
|
305
414
|
}
|
|
306
|
-
const request = getRequestObject(triggerType, auth, null, null, config);
|
|
307
|
-
|
|
308
|
-
|
|
309
|
-
|
|
310
|
-
|
|
415
|
+
const request = getRequestObject(triggerType, auth, null, null, config, context, isGet);
|
|
416
|
+
// Convert query parameter to Parse.Query instance
|
|
417
|
+
if (query instanceof _node.default.Query) {
|
|
418
|
+
request.query = query;
|
|
419
|
+
} else if (typeof query === 'object' && query !== null) {
|
|
420
|
+
const parseQueryInstance = new _node.default.Query(classNameQuery);
|
|
421
|
+
if (query.where) {
|
|
422
|
+
parseQueryInstance.withJSON(query);
|
|
423
|
+
}
|
|
424
|
+
request.query = parseQueryInstance;
|
|
425
|
+
} else {
|
|
426
|
+
request.query = new _node.default.Query(classNameQuery);
|
|
427
|
+
}
|
|
428
|
+
const {
|
|
429
|
+
success,
|
|
430
|
+
error
|
|
431
|
+
} = getResponseObject(request, processedObjectsJSON => {
|
|
432
|
+
resolve(processedObjectsJSON);
|
|
433
|
+
}, errorData => {
|
|
434
|
+
reject(errorData);
|
|
311
435
|
});
|
|
312
|
-
logTriggerSuccessBeforeHook(triggerType,
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
436
|
+
logTriggerSuccessBeforeHook(triggerType, classNameQuery, 'AfterFind Input (Pre-Transform)', JSON.stringify(objectsInput.map(o => o instanceof _node.default.Object ? o.id + ':' + o.className : o)), auth, config.logLevels.triggerBeforeSuccess);
|
|
437
|
+
|
|
438
|
+
// Convert plain objects to Parse.Object instances for trigger
|
|
439
|
+
request.objects = objectsInput.map(currentObject => {
|
|
440
|
+
if (currentObject instanceof _node.default.Object) {
|
|
441
|
+
return currentObject;
|
|
442
|
+
}
|
|
443
|
+
// Preserve the original className if it exists, otherwise use the query className
|
|
444
|
+
const originalClassName = currentObject.className || classNameQuery;
|
|
445
|
+
const tempObjectWithClassName = {
|
|
446
|
+
...currentObject,
|
|
447
|
+
className: originalClassName
|
|
448
|
+
};
|
|
449
|
+
return _node.default.Object.fromJSON(tempObjectWithClassName);
|
|
317
450
|
});
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
451
|
+
return Promise.resolve().then(() => {
|
|
452
|
+
return maybeRunValidator(request, `${triggerType}.${classNameQuery}`, auth);
|
|
453
|
+
}).then(() => {
|
|
454
|
+
if (request.skipWithMasterKey) {
|
|
455
|
+
return request.objects;
|
|
456
|
+
}
|
|
457
|
+
const responseFromTrigger = trigger(request);
|
|
458
|
+
if (responseFromTrigger && typeof responseFromTrigger.then === 'function') {
|
|
459
|
+
return responseFromTrigger.then(results => {
|
|
460
|
+
return results;
|
|
461
|
+
});
|
|
462
|
+
}
|
|
463
|
+
return responseFromTrigger;
|
|
464
|
+
}).then(success, error);
|
|
465
|
+
}).then(resultsAsJSON => {
|
|
466
|
+
logTriggerAfterHook(triggerType, classNameQuery, JSON.stringify(resultsAsJSON), auth, config.logLevels.triggerAfter);
|
|
467
|
+
return resultsAsJSON;
|
|
331
468
|
});
|
|
332
469
|
}
|
|
333
|
-
|
|
334
|
-
function maybeRunQueryTrigger(triggerType, className, restWhere, restOptions, config, auth, isGet) {
|
|
470
|
+
function maybeRunQueryTrigger(triggerType, className, restWhere, restOptions, config, auth, context, isGet) {
|
|
335
471
|
const trigger = getTrigger(className, triggerType, config.applicationId);
|
|
336
472
|
if (!trigger) {
|
|
337
473
|
return Promise.resolve({
|
|
@@ -339,30 +475,25 @@ function maybeRunQueryTrigger(triggerType, className, restWhere, restOptions, co
|
|
|
339
475
|
restOptions
|
|
340
476
|
});
|
|
341
477
|
}
|
|
342
|
-
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
|
|
346
|
-
}
|
|
478
|
+
const json = Object.assign({}, restOptions);
|
|
479
|
+
json.where = restWhere;
|
|
480
|
+
const parseQuery = new _node.default.Query(className);
|
|
481
|
+
parseQuery.withJSON(json);
|
|
347
482
|
let count = false;
|
|
348
483
|
if (restOptions) {
|
|
349
|
-
if (restOptions.include && restOptions.include.length > 0) {
|
|
350
|
-
parseQuery._include = restOptions.include.split(',');
|
|
351
|
-
}
|
|
352
|
-
if (restOptions.skip) {
|
|
353
|
-
parseQuery._skip = restOptions.skip;
|
|
354
|
-
}
|
|
355
|
-
if (restOptions.limit) {
|
|
356
|
-
parseQuery._limit = restOptions.limit;
|
|
357
|
-
}
|
|
358
484
|
count = !!restOptions.count;
|
|
359
485
|
}
|
|
360
|
-
const requestObject = getRequestQueryObject(triggerType, auth, parseQuery, count, config, isGet);
|
|
486
|
+
const requestObject = getRequestQueryObject(triggerType, auth, parseQuery, count, config, context, isGet);
|
|
361
487
|
return Promise.resolve().then(() => {
|
|
488
|
+
return maybeRunValidator(requestObject, `${triggerType}.${className}`, auth);
|
|
489
|
+
}).then(() => {
|
|
490
|
+
if (requestObject.skipWithMasterKey) {
|
|
491
|
+
return requestObject.query;
|
|
492
|
+
}
|
|
362
493
|
return trigger(requestObject);
|
|
363
494
|
}).then(result => {
|
|
364
495
|
let queryResult = parseQuery;
|
|
365
|
-
if (result && result instanceof
|
|
496
|
+
if (result && result instanceof _node.default.Query) {
|
|
366
497
|
queryResult = result;
|
|
367
498
|
}
|
|
368
499
|
const jsonQuery = queryResult.toJSON();
|
|
@@ -381,6 +512,14 @@ function maybeRunQueryTrigger(triggerType, className, restWhere, restOptions, co
|
|
|
381
512
|
restOptions = restOptions || {};
|
|
382
513
|
restOptions.include = jsonQuery.include;
|
|
383
514
|
}
|
|
515
|
+
if (jsonQuery.excludeKeys) {
|
|
516
|
+
restOptions = restOptions || {};
|
|
517
|
+
restOptions.excludeKeys = jsonQuery.excludeKeys;
|
|
518
|
+
}
|
|
519
|
+
if (jsonQuery.explain) {
|
|
520
|
+
restOptions = restOptions || {};
|
|
521
|
+
restOptions.explain = jsonQuery.explain;
|
|
522
|
+
}
|
|
384
523
|
if (jsonQuery.keys) {
|
|
385
524
|
restOptions = restOptions || {};
|
|
386
525
|
restOptions.keys = jsonQuery.keys;
|
|
@@ -389,6 +528,14 @@ function maybeRunQueryTrigger(triggerType, className, restWhere, restOptions, co
|
|
|
389
528
|
restOptions = restOptions || {};
|
|
390
529
|
restOptions.order = jsonQuery.order;
|
|
391
530
|
}
|
|
531
|
+
if (jsonQuery.hint) {
|
|
532
|
+
restOptions = restOptions || {};
|
|
533
|
+
restOptions.hint = jsonQuery.hint;
|
|
534
|
+
}
|
|
535
|
+
if (jsonQuery.comment) {
|
|
536
|
+
restOptions = restOptions || {};
|
|
537
|
+
restOptions.comment = jsonQuery.comment;
|
|
538
|
+
}
|
|
392
539
|
if (requestObject.readPreference) {
|
|
393
540
|
restOptions = restOptions || {};
|
|
394
541
|
restOptions.readPreference = requestObject.readPreference;
|
|
@@ -401,75 +548,370 @@ function maybeRunQueryTrigger(triggerType, className, restWhere, restOptions, co
|
|
|
401
548
|
restOptions = restOptions || {};
|
|
402
549
|
restOptions.subqueryReadPreference = requestObject.subqueryReadPreference;
|
|
403
550
|
}
|
|
551
|
+
let objects = undefined;
|
|
552
|
+
if (result instanceof _node.default.Object) {
|
|
553
|
+
objects = [result];
|
|
554
|
+
} else if (Array.isArray(result) && (!result.length || result.every(obj => obj instanceof _node.default.Object))) {
|
|
555
|
+
objects = result;
|
|
556
|
+
}
|
|
404
557
|
return {
|
|
405
558
|
restWhere,
|
|
406
|
-
restOptions
|
|
559
|
+
restOptions,
|
|
560
|
+
objects
|
|
407
561
|
};
|
|
408
562
|
}, err => {
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
563
|
+
const error = resolveError(err, {
|
|
564
|
+
code: _node.default.Error.SCRIPT_FAILED,
|
|
565
|
+
message: 'Script failed. Unknown error.'
|
|
566
|
+
});
|
|
567
|
+
throw error;
|
|
414
568
|
});
|
|
415
569
|
}
|
|
570
|
+
function resolveError(message, defaultOpts) {
|
|
571
|
+
if (!defaultOpts) {
|
|
572
|
+
defaultOpts = {};
|
|
573
|
+
}
|
|
574
|
+
if (!message) {
|
|
575
|
+
return new _node.default.Error(defaultOpts.code || _node.default.Error.SCRIPT_FAILED, defaultOpts.message || 'Script failed.');
|
|
576
|
+
}
|
|
577
|
+
if (message instanceof _node.default.Error) {
|
|
578
|
+
return message;
|
|
579
|
+
}
|
|
580
|
+
const code = defaultOpts.code || _node.default.Error.SCRIPT_FAILED;
|
|
581
|
+
// If it's an error, mark it as a script failed
|
|
582
|
+
if (typeof message === 'string') {
|
|
583
|
+
return new _node.default.Error(code, message);
|
|
584
|
+
}
|
|
585
|
+
const error = new _node.default.Error(code, message.message || message);
|
|
586
|
+
if (message instanceof Error) {
|
|
587
|
+
error.stack = message.stack;
|
|
588
|
+
}
|
|
589
|
+
return error;
|
|
590
|
+
}
|
|
591
|
+
function maybeRunValidator(request, functionName, auth) {
|
|
592
|
+
const theValidator = getValidator(functionName, _node.default.applicationId);
|
|
593
|
+
if (!theValidator) {
|
|
594
|
+
return;
|
|
595
|
+
}
|
|
596
|
+
if (typeof theValidator === 'object' && theValidator.skipWithMasterKey && request.master) {
|
|
597
|
+
request.skipWithMasterKey = true;
|
|
598
|
+
}
|
|
599
|
+
return new Promise((resolve, reject) => {
|
|
600
|
+
return Promise.resolve().then(() => {
|
|
601
|
+
return typeof theValidator === 'object' ? builtInTriggerValidator(theValidator, request, auth) : theValidator(request);
|
|
602
|
+
}).then(() => {
|
|
603
|
+
resolve();
|
|
604
|
+
}).catch(e => {
|
|
605
|
+
const error = resolveError(e, {
|
|
606
|
+
code: _node.default.Error.VALIDATION_ERROR,
|
|
607
|
+
message: 'Validation failed.'
|
|
608
|
+
});
|
|
609
|
+
reject(error);
|
|
610
|
+
});
|
|
611
|
+
});
|
|
612
|
+
}
|
|
613
|
+
async function builtInTriggerValidator(options, request, auth) {
|
|
614
|
+
if (request.master && !options.validateMasterKey) {
|
|
615
|
+
return;
|
|
616
|
+
}
|
|
617
|
+
let reqUser = request.user;
|
|
618
|
+
if (!reqUser && request.object && request.object.className === '_User' && !request.object.existed()) {
|
|
619
|
+
reqUser = request.object;
|
|
620
|
+
}
|
|
621
|
+
if ((options.requireUser || options.requireAnyUserRoles || options.requireAllUserRoles) && !reqUser) {
|
|
622
|
+
throw 'Validation failed. Please login to continue.';
|
|
623
|
+
}
|
|
624
|
+
if (options.requireMaster && !request.master) {
|
|
625
|
+
throw 'Validation failed. Master key is required to complete this request.';
|
|
626
|
+
}
|
|
627
|
+
let params = request.params || {};
|
|
628
|
+
if (request.object) {
|
|
629
|
+
params = request.object.toJSON();
|
|
630
|
+
}
|
|
631
|
+
const requiredParam = key => {
|
|
632
|
+
const value = params[key];
|
|
633
|
+
if (value == null) {
|
|
634
|
+
throw `Validation failed. Please specify data for ${key}.`;
|
|
635
|
+
}
|
|
636
|
+
};
|
|
637
|
+
const validateOptions = async (opt, key, val) => {
|
|
638
|
+
let opts = opt.options;
|
|
639
|
+
if (typeof opts === 'function') {
|
|
640
|
+
try {
|
|
641
|
+
const result = await opts(val);
|
|
642
|
+
if (!result && result != null) {
|
|
643
|
+
throw opt.error || `Validation failed. Invalid value for ${key}.`;
|
|
644
|
+
}
|
|
645
|
+
} catch (e) {
|
|
646
|
+
if (!e) {
|
|
647
|
+
throw opt.error || `Validation failed. Invalid value for ${key}.`;
|
|
648
|
+
}
|
|
649
|
+
throw opt.error || e.message || e;
|
|
650
|
+
}
|
|
651
|
+
return;
|
|
652
|
+
}
|
|
653
|
+
if (!Array.isArray(opts)) {
|
|
654
|
+
opts = [opt.options];
|
|
655
|
+
}
|
|
656
|
+
if (!opts.includes(val)) {
|
|
657
|
+
throw opt.error || `Validation failed. Invalid option for ${key}. Expected: ${opts.join(', ')}`;
|
|
658
|
+
}
|
|
659
|
+
};
|
|
660
|
+
const getType = fn => {
|
|
661
|
+
const match = fn && fn.toString().match(/^\s*function (\w+)/);
|
|
662
|
+
return (match ? match[1] : '').toLowerCase();
|
|
663
|
+
};
|
|
664
|
+
if (Array.isArray(options.fields)) {
|
|
665
|
+
for (const key of options.fields) {
|
|
666
|
+
requiredParam(key);
|
|
667
|
+
}
|
|
668
|
+
} else {
|
|
669
|
+
const optionPromises = [];
|
|
670
|
+
for (const key in options.fields) {
|
|
671
|
+
const opt = options.fields[key];
|
|
672
|
+
let val = params[key];
|
|
673
|
+
if (typeof opt === 'string') {
|
|
674
|
+
requiredParam(opt);
|
|
675
|
+
}
|
|
676
|
+
if (typeof opt === 'object') {
|
|
677
|
+
if (opt.default != null && val == null) {
|
|
678
|
+
val = opt.default;
|
|
679
|
+
params[key] = val;
|
|
680
|
+
if (request.object) {
|
|
681
|
+
request.object.set(key, val);
|
|
682
|
+
}
|
|
683
|
+
}
|
|
684
|
+
if (opt.constant && request.object) {
|
|
685
|
+
if (request.original) {
|
|
686
|
+
request.object.revert(key);
|
|
687
|
+
} else if (opt.default != null) {
|
|
688
|
+
request.object.set(key, opt.default);
|
|
689
|
+
}
|
|
690
|
+
}
|
|
691
|
+
if (opt.required) {
|
|
692
|
+
requiredParam(key);
|
|
693
|
+
}
|
|
694
|
+
const optional = !opt.required && val === undefined;
|
|
695
|
+
if (!optional) {
|
|
696
|
+
if (opt.type) {
|
|
697
|
+
const type = getType(opt.type);
|
|
698
|
+
const valType = Array.isArray(val) ? 'array' : typeof val;
|
|
699
|
+
if (valType !== type) {
|
|
700
|
+
throw `Validation failed. Invalid type for ${key}. Expected: ${type}`;
|
|
701
|
+
}
|
|
702
|
+
}
|
|
703
|
+
if (opt.options) {
|
|
704
|
+
optionPromises.push(validateOptions(opt, key, val));
|
|
705
|
+
}
|
|
706
|
+
}
|
|
707
|
+
}
|
|
708
|
+
}
|
|
709
|
+
await Promise.all(optionPromises);
|
|
710
|
+
}
|
|
711
|
+
let userRoles = options.requireAnyUserRoles;
|
|
712
|
+
let requireAllRoles = options.requireAllUserRoles;
|
|
713
|
+
const promises = [Promise.resolve(), Promise.resolve(), Promise.resolve()];
|
|
714
|
+
if (userRoles || requireAllRoles) {
|
|
715
|
+
promises[0] = auth.getUserRoles();
|
|
716
|
+
}
|
|
717
|
+
if (typeof userRoles === 'function') {
|
|
718
|
+
promises[1] = userRoles();
|
|
719
|
+
}
|
|
720
|
+
if (typeof requireAllRoles === 'function') {
|
|
721
|
+
promises[2] = requireAllRoles();
|
|
722
|
+
}
|
|
723
|
+
const [roles, resolvedUserRoles, resolvedRequireAll] = await Promise.all(promises);
|
|
724
|
+
if (resolvedUserRoles && Array.isArray(resolvedUserRoles)) {
|
|
725
|
+
userRoles = resolvedUserRoles;
|
|
726
|
+
}
|
|
727
|
+
if (resolvedRequireAll && Array.isArray(resolvedRequireAll)) {
|
|
728
|
+
requireAllRoles = resolvedRequireAll;
|
|
729
|
+
}
|
|
730
|
+
if (userRoles) {
|
|
731
|
+
const hasRole = userRoles.some(requiredRole => roles.includes(`role:${requiredRole}`));
|
|
732
|
+
if (!hasRole) {
|
|
733
|
+
throw `Validation failed. User does not match the required roles.`;
|
|
734
|
+
}
|
|
735
|
+
}
|
|
736
|
+
if (requireAllRoles) {
|
|
737
|
+
for (const requiredRole of requireAllRoles) {
|
|
738
|
+
if (!roles.includes(`role:${requiredRole}`)) {
|
|
739
|
+
throw `Validation failed. User does not match all the required roles.`;
|
|
740
|
+
}
|
|
741
|
+
}
|
|
742
|
+
}
|
|
743
|
+
const userKeys = options.requireUserKeys || [];
|
|
744
|
+
if (Array.isArray(userKeys)) {
|
|
745
|
+
for (const key of userKeys) {
|
|
746
|
+
if (!reqUser) {
|
|
747
|
+
throw 'Please login to make this request.';
|
|
748
|
+
}
|
|
749
|
+
if (reqUser.get(key) == null) {
|
|
750
|
+
throw `Validation failed. Please set data for ${key} on your account.`;
|
|
751
|
+
}
|
|
752
|
+
}
|
|
753
|
+
} else if (typeof userKeys === 'object') {
|
|
754
|
+
const optionPromises = [];
|
|
755
|
+
for (const key in options.requireUserKeys) {
|
|
756
|
+
const opt = options.requireUserKeys[key];
|
|
757
|
+
if (opt.options) {
|
|
758
|
+
optionPromises.push(validateOptions(opt, key, reqUser.get(key)));
|
|
759
|
+
}
|
|
760
|
+
}
|
|
761
|
+
await Promise.all(optionPromises);
|
|
762
|
+
}
|
|
763
|
+
}
|
|
416
764
|
|
|
417
765
|
// To be used as part of the promise chain when saving/deleting an object
|
|
418
766
|
// Will resolve successfully if no trigger is configured
|
|
419
767
|
// Resolves to an object, empty or containing an object key. A beforeSave
|
|
420
768
|
// trigger will set the object key to the rest format object to save.
|
|
421
769
|
// originalParseObject is optional, we only need that for before/afterSave functions
|
|
422
|
-
function maybeRunTrigger(triggerType, auth, parseObject, originalParseObject, config) {
|
|
770
|
+
function maybeRunTrigger(triggerType, auth, parseObject, originalParseObject, config, context) {
|
|
423
771
|
if (!parseObject) {
|
|
424
772
|
return Promise.resolve({});
|
|
425
773
|
}
|
|
426
774
|
return new Promise(function (resolve, reject) {
|
|
427
775
|
var trigger = getTrigger(parseObject.className, triggerType, config.applicationId);
|
|
428
|
-
if (!trigger)
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
|
|
776
|
+
if (!trigger) {
|
|
777
|
+
return resolve();
|
|
778
|
+
}
|
|
779
|
+
var request = getRequestObject(triggerType, auth, parseObject, originalParseObject, config, context);
|
|
780
|
+
var {
|
|
781
|
+
success,
|
|
782
|
+
error
|
|
783
|
+
} = getResponseObject(request, object => {
|
|
784
|
+
logTriggerSuccessBeforeHook(triggerType, parseObject.className, parseObject.toJSON(), object, auth, triggerType.startsWith('after') ? config.logLevels.triggerAfter : config.logLevels.triggerBeforeSuccess);
|
|
785
|
+
if (triggerType === Types.beforeSave || triggerType === Types.afterSave || triggerType === Types.beforeDelete || triggerType === Types.afterDelete) {
|
|
786
|
+
Object.assign(context, request.context);
|
|
787
|
+
}
|
|
432
788
|
resolve(object);
|
|
433
789
|
}, error => {
|
|
434
|
-
logTriggerErrorBeforeHook(triggerType, parseObject.className, parseObject.toJSON(), auth, error);
|
|
790
|
+
logTriggerErrorBeforeHook(triggerType, parseObject.className, parseObject.toJSON(), auth, error, config.logLevels.triggerBeforeError);
|
|
435
791
|
reject(error);
|
|
436
792
|
});
|
|
437
|
-
// Force the current Parse app before the trigger
|
|
438
|
-
_node2.default.applicationId = config.applicationId;
|
|
439
|
-
_node2.default.javascriptKey = config.javascriptKey || '';
|
|
440
|
-
_node2.default.masterKey = config.masterKey;
|
|
441
793
|
|
|
442
794
|
// AfterSave and afterDelete triggers can return a promise, which if they
|
|
443
795
|
// do, needs to be resolved before this promise is resolved,
|
|
444
796
|
// so trigger execution is synced with RestWrite.execute() call.
|
|
445
797
|
// If triggers do not return a promise, they can run async code parallel
|
|
446
798
|
// to the RestWrite.execute() call.
|
|
447
|
-
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
if (
|
|
451
|
-
return
|
|
452
|
-
} else {
|
|
453
|
-
return resolve();
|
|
799
|
+
return Promise.resolve().then(() => {
|
|
800
|
+
return maybeRunValidator(request, `${triggerType}.${parseObject.className}`, auth);
|
|
801
|
+
}).then(() => {
|
|
802
|
+
if (request.skipWithMasterKey) {
|
|
803
|
+
return Promise.resolve();
|
|
454
804
|
}
|
|
455
|
-
|
|
805
|
+
const promise = trigger(request);
|
|
806
|
+
if (triggerType === Types.afterSave || triggerType === Types.afterDelete || triggerType === Types.afterLogin) {
|
|
807
|
+
logTriggerAfterHook(triggerType, parseObject.className, parseObject.toJSON(), auth, config.logLevels.triggerAfter);
|
|
808
|
+
}
|
|
809
|
+
// beforeSave is expected to return null (nothing)
|
|
810
|
+
if (triggerType === Types.beforeSave) {
|
|
811
|
+
if (promise && typeof promise.then === 'function') {
|
|
812
|
+
return promise.then(response => {
|
|
813
|
+
// response.object may come from express routing before hook
|
|
814
|
+
if (response && response.object) {
|
|
815
|
+
return response;
|
|
816
|
+
}
|
|
817
|
+
return null;
|
|
818
|
+
});
|
|
819
|
+
}
|
|
820
|
+
return null;
|
|
821
|
+
}
|
|
822
|
+
return promise;
|
|
823
|
+
}).then(success, error);
|
|
456
824
|
});
|
|
457
825
|
}
|
|
458
826
|
|
|
459
827
|
// Converts a REST-format object to a Parse.Object
|
|
460
828
|
// data is either className or an object
|
|
461
829
|
function inflate(data, restObject) {
|
|
462
|
-
var copy = typeof data == 'object' ? data : {
|
|
830
|
+
var copy = typeof data == 'object' ? data : {
|
|
831
|
+
className: data
|
|
832
|
+
};
|
|
463
833
|
for (var key in restObject) {
|
|
464
834
|
copy[key] = restObject[key];
|
|
465
835
|
}
|
|
466
|
-
return
|
|
836
|
+
return _node.default.Object.fromJSON(copy);
|
|
467
837
|
}
|
|
468
|
-
|
|
469
|
-
function runLiveQueryEventHandlers(data, applicationId = _node2.default.applicationId) {
|
|
838
|
+
function runLiveQueryEventHandlers(data, applicationId = _node.default.applicationId) {
|
|
470
839
|
if (!_triggerStore || !_triggerStore[applicationId] || !_triggerStore[applicationId].LiveQuery) {
|
|
471
840
|
return;
|
|
472
841
|
}
|
|
473
842
|
_triggerStore[applicationId].LiveQuery.forEach(handler => handler(data));
|
|
474
843
|
}
|
|
475
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/triggers.js"],"names":["addFunction","addJob","addTrigger","addLiveQueryEventHandler","removeFunction","removeTrigger","_unregisterAll","getTrigger","triggerExists","getFunction","getJob","getJobs","getValidator","getRequestObject","getRequestQueryObject","getResponseObject","maybeRunAfterFindTrigger","maybeRunQueryTrigger","maybeRunTrigger","inflate","runLiveQueryEventHandlers","Types","beforeSave","afterSave","beforeDelete","afterDelete","beforeFind","afterFind","baseStore","Validators","Functions","Jobs","LiveQuery","Triggers","Object","keys","reduce","base","key","freeze","validateClassNameForTriggers","className","type","restrictedClassNames","indexOf","_triggerStore","functionName","handler","validationHandler","applicationId","Parse","jobName","push","forEach","appId","triggerType","manager","undefined","auth","parseObject","originalParseObject","config","request","triggerName","object","master","log","loggerController","headers","ip","original","isMaster","user","installationId","query","count","isGet","resolve","reject","success","response","objects","map","toJSON","equals","_getSaveJSON","error","code","message","Error","SCRIPT_FAILED","scriptError","userIdForLog","id","logTriggerAfterHook","input","cleanInput","logger","truncateLogMessage","JSON","stringify","info","logTriggerSuccessBeforeHook","result","cleanResult","logTriggerErrorBeforeHook","Promise","trigger","fromJSON","triggerPromise","then","promiseResults","results","restWhere","restOptions","parseQuery","Query","_where","include","length","_include","split","skip","_skip","limit","_limit","requestObject","queryResult","jsonQuery","where","order","readPreference","includeReadPreference","subqueryReadPreference","err","javascriptKey","masterKey","data","restObject","copy"],"mappings":"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","file":"triggers.js","sourcesContent":["// triggers.js\nimport Parse    from 'parse/node';\nimport { logger } from './logger';\n\nexport const Types = {\n  beforeSave: 'beforeSave',\n  afterSave: 'afterSave',\n  beforeDelete: 'beforeDelete',\n  afterDelete: 'afterDelete',\n  beforeFind: 'beforeFind',\n  afterFind: 'afterFind'\n};\n\nconst baseStore = function() {\n  const Validators = {};\n  const Functions = {};\n  const Jobs = {};\n  const LiveQuery = [];\n  const Triggers = Object.keys(Types).reduce(function(base, key){\n    base[key] = {};\n    return base;\n  }, {});\n\n  return Object.freeze({\n    Functions,\n    Jobs,\n    Validators,\n    Triggers,\n    LiveQuery,\n  });\n};\n\nfunction validateClassNameForTriggers(className, type) {\n  const restrictedClassNames = [ '_Session' ];\n  if (restrictedClassNames.indexOf(className) != -1) {\n    throw `Triggers are not supported for ${className} class.`;\n  }\n  if (type == Types.beforeSave && className === '_PushStatus') {\n    // _PushStatus uses undocumented nested key increment ops\n    // allowing beforeSave would mess up the objects big time\n    // TODO: Allow proper documented way of using nested increment ops\n    throw 'Only afterSave is allowed on _PushStatus';\n  }\n  return className;\n}\n\nconst _triggerStore = {};\n\nexport function addFunction(functionName, handler, validationHandler, applicationId) {\n  applicationId = applicationId || Parse.applicationId;\n  _triggerStore[applicationId] =  _triggerStore[applicationId] || baseStore();\n  _triggerStore[applicationId].Functions[functionName] = handler;\n  _triggerStore[applicationId].Validators[functionName] = validationHandler;\n}\n\nexport function addJob(jobName, handler, applicationId) {\n  applicationId = applicationId || Parse.applicationId;\n  _triggerStore[applicationId] =  _triggerStore[applicationId] || baseStore();\n  _triggerStore[applicationId].Jobs[jobName] = handler;\n}\n\nexport function addTrigger(type, className, handler, applicationId) {\n  validateClassNameForTriggers(className, type);\n  applicationId = applicationId || Parse.applicationId;\n  _triggerStore[applicationId] =  _triggerStore[applicationId] || baseStore();\n  _triggerStore[applicationId].Triggers[type][className] = handler;\n}\n\nexport function addLiveQueryEventHandler(handler, applicationId) {\n  applicationId = applicationId || Parse.applicationId;\n  _triggerStore[applicationId] =  _triggerStore[applicationId] || baseStore();\n  _triggerStore[applicationId].LiveQuery.push(handler);\n}\n\nexport function removeFunction(functionName, applicationId) {\n  applicationId = applicationId || Parse.applicationId;\n  delete _triggerStore[applicationId].Functions[functionName]\n}\n\nexport function removeTrigger(type, className, applicationId) {\n  applicationId = applicationId || Parse.applicationId;\n  delete _triggerStore[applicationId].Triggers[type][className]\n}\n\nexport function _unregisterAll() {\n  Object.keys(_triggerStore).forEach(appId => delete _triggerStore[appId]);\n}\n\nexport function getTrigger(className, triggerType, applicationId) {\n  if (!applicationId) {\n    throw \"Missing ApplicationID\";\n  }\n  var manager = _triggerStore[applicationId]\n  if (manager\n    && manager.Triggers\n    && manager.Triggers[triggerType]\n    && manager.Triggers[triggerType][className]) {\n    return manager.Triggers[triggerType][className];\n  }\n  return undefined;\n}\n\nexport function triggerExists(className: string, type: string, applicationId: string): boolean {\n  return (getTrigger(className, type, applicationId) != undefined);\n}\n\nexport function getFunction(functionName, applicationId) {\n  var manager = _triggerStore[applicationId];\n  if (manager && manager.Functions) {\n    return manager.Functions[functionName];\n  }\n  return undefined;\n}\n\nexport function getJob(jobName, applicationId) {\n  var manager = _triggerStore[applicationId];\n  if (manager && manager.Jobs) {\n    return manager.Jobs[jobName];\n  }\n  return undefined;\n}\n\nexport function getJobs(applicationId) {\n  var manager = _triggerStore[applicationId];\n  if (manager && manager.Jobs) {\n    return manager.Jobs;\n  }\n  return undefined;\n}\n\n\nexport function getValidator(functionName, applicationId) {\n  var manager = _triggerStore[applicationId];\n  if (manager && manager.Validators) {\n    return manager.Validators[functionName];\n  }\n  return undefined;\n}\n\nexport function getRequestObject(triggerType, auth, parseObject, originalParseObject, config) {\n  var request = {\n    triggerName: triggerType,\n    object: parseObject,\n    master: false,\n    log: config.loggerController,\n    headers: config.headers,\n    ip: config.ip,\n  };\n\n  if (originalParseObject) {\n    request.original = originalParseObject;\n  }\n\n  if (!auth) {\n    return request;\n  }\n  if (auth.isMaster) {\n    request['master'] = true;\n  }\n  if (auth.user) {\n    request['user'] = auth.user;\n  }\n  if (auth.installationId) {\n    request['installationId'] = auth.installationId;\n  }\n  return request;\n}\n\nexport function getRequestQueryObject(triggerType, auth, query, count, config, isGet) {\n  isGet = !!isGet;\n\n  var request = {\n    triggerName: triggerType,\n    query,\n    master: false,\n    count,\n    log: config.loggerController,\n    isGet,\n    headers: config.headers,\n    ip: config.ip,\n  };\n\n  if (!auth) {\n    return request;\n  }\n  if (auth.isMaster) {\n    request['master'] = true;\n  }\n  if (auth.user) {\n    request['user'] = auth.user;\n  }\n  if (auth.installationId) {\n    request['installationId'] = auth.installationId;\n  }\n  return request;\n}\n\n// Creates the response object, and uses the request object to pass data\n// The API will call this with REST API formatted objects, this will\n// transform them to Parse.Object instances expected by Cloud Code.\n// Any changes made to the object in a beforeSave will be included.\nexport function getResponseObject(request, resolve, reject) {\n  return {\n    success: function(response) {\n      if (request.triggerName === Types.afterFind) {\n        if(!response){\n          response = request.objects;\n        }\n        response = response.map(object => {\n          return object.toJSON();\n        });\n        return resolve(response);\n      }\n      // Use the JSON response\n      if (response && !request.object.equals(response)\n          && request.triggerName === Types.beforeSave) {\n        return resolve(response);\n      }\n      response = {};\n      if (request.triggerName === Types.beforeSave) {\n        response['object'] = request.object._getSaveJSON();\n      }\n      return resolve(response);\n    },\n    error: function(code, message) {\n      if (!message) {\n        if (code instanceof Parse.Error) {\n          return reject(code)\n        }\n        message = code;\n        code = Parse.Error.SCRIPT_FAILED;\n      }\n      var scriptError = new Parse.Error(code, message);\n      return reject(scriptError);\n    }\n  }\n}\n\nfunction userIdForLog(auth) {\n  return (auth && auth.user) ? auth.user.id : undefined;\n}\n\nfunction logTriggerAfterHook(triggerType, className, input, auth) {\n  const cleanInput = logger.truncateLogMessage(JSON.stringify(input));\n  logger.info(`${triggerType} triggered for ${className} for user ${userIdForLog(auth)}:\\n  Input: ${cleanInput}`, {\n    className,\n    triggerType,\n    user: userIdForLog(auth)\n  });\n}\n\nfunction logTriggerSuccessBeforeHook(triggerType, className, input, result, auth) {\n  const cleanInput = logger.truncateLogMessage(JSON.stringify(input));\n  const cleanResult = logger.truncateLogMessage(JSON.stringify(result));\n  logger.info(`${triggerType} triggered for ${className} for user ${userIdForLog(auth)}:\\n  Input: ${cleanInput}\\n  Result: ${cleanResult}`, {\n    className,\n    triggerType,\n    user: userIdForLog(auth)\n  });\n}\n\nfunction logTriggerErrorBeforeHook(triggerType, className, input, auth, error) {\n  const cleanInput = logger.truncateLogMessage(JSON.stringify(input));\n  logger.error(`${triggerType} failed for ${className} for user ${userIdForLog(auth)}:\\n  Input: ${cleanInput}\\n  Error: ${JSON.stringify(error)}`, {\n    className,\n    triggerType,\n    error,\n    user: userIdForLog(auth)\n  });\n}\n\nexport function maybeRunAfterFindTrigger(triggerType, auth, className, objects, config) {\n  return new Promise((resolve, reject) => {\n    const trigger = getTrigger(className, triggerType, config.applicationId);\n    if (!trigger) {\n      return resolve();\n    }\n    const request = getRequestObject(triggerType, auth, null, null, config);\n    const response = getResponseObject(request,\n      object => {\n        resolve(object);\n      },\n      error => {\n        reject(error);\n      });\n    logTriggerSuccessBeforeHook(triggerType, className, 'AfterFind', JSON.stringify(objects), auth);\n    request.objects = objects.map(object => {\n      //setting the class name to transform into parse object\n      object.className = className;\n      return Parse.Object.fromJSON(object);\n    });\n    const triggerPromise = trigger(request, response);\n    if (triggerPromise && typeof triggerPromise.then === \"function\") {\n      return triggerPromise.then(promiseResults => {\n        if(promiseResults) {\n          resolve(promiseResults);\n        }else{\n          return reject(new Parse.Error(Parse.Error.SCRIPT_FAILED, \"AfterFind expect results to be returned in the promise\"));\n        }\n      });\n    }\n  }).then((results) => {\n    logTriggerAfterHook(triggerType, className, JSON.stringify(results), auth);\n    return results;\n  });\n}\n\nexport function maybeRunQueryTrigger(triggerType, className, restWhere, restOptions, config, auth, isGet) {\n  const trigger = getTrigger(className, triggerType, config.applicationId);\n  if (!trigger) {\n    return Promise.resolve({\n      restWhere,\n      restOptions\n    });\n  }\n\n  const parseQuery = new Parse.Query(className);\n  if (restWhere) {\n    parseQuery._where = restWhere;\n  }\n  let count = false;\n  if (restOptions) {\n    if (restOptions.include && restOptions.include.length > 0) {\n      parseQuery._include = restOptions.include.split(',');\n    }\n    if (restOptions.skip) {\n      parseQuery._skip = restOptions.skip;\n    }\n    if (restOptions.limit) {\n      parseQuery._limit = restOptions.limit;\n    }\n    count = !!restOptions.count;\n  }\n  const requestObject = getRequestQueryObject(triggerType, auth, parseQuery, count, config, isGet);\n  return Promise.resolve().then(() => {\n    return trigger(requestObject);\n  }).then((result) => {\n    let queryResult = parseQuery;\n    if (result && result instanceof Parse.Query) {\n      queryResult = result;\n    }\n    const jsonQuery = queryResult.toJSON();\n    if (jsonQuery.where) {\n      restWhere = jsonQuery.where;\n    }\n    if (jsonQuery.limit) {\n      restOptions = restOptions || {};\n      restOptions.limit = jsonQuery.limit;\n    }\n    if (jsonQuery.skip) {\n      restOptions = restOptions || {};\n      restOptions.skip = jsonQuery.skip;\n    }\n    if (jsonQuery.include) {\n      restOptions = restOptions || {};\n      restOptions.include = jsonQuery.include;\n    }\n    if (jsonQuery.keys) {\n      restOptions = restOptions || {};\n      restOptions.keys = jsonQuery.keys;\n    }\n    if (jsonQuery.order) {\n      restOptions = restOptions || {};\n      restOptions.order = jsonQuery.order;\n    }\n    if (requestObject.readPreference) {\n      restOptions = restOptions || {};\n      restOptions.readPreference = requestObject.readPreference;\n    }\n    if (requestObject.includeReadPreference) {\n      restOptions = restOptions || {};\n      restOptions.includeReadPreference = requestObject.includeReadPreference;\n    }\n    if (requestObject.subqueryReadPreference) {\n      restOptions = restOptions || {};\n      restOptions.subqueryReadPreference = requestObject.subqueryReadPreference;\n    }\n    return {\n      restWhere,\n      restOptions\n    };\n  }, (err) => {\n    if (typeof err === 'string') {\n      throw new Parse.Error(1, err);\n    } else {\n      throw err;\n    }\n  });\n}\n\n// To be used as part of the promise chain when saving/deleting an object\n// Will resolve successfully if no trigger is configured\n// Resolves to an object, empty or containing an object key. A beforeSave\n// trigger will set the object key to the rest format object to save.\n// originalParseObject is optional, we only need that for before/afterSave functions\nexport function maybeRunTrigger(triggerType, auth, parseObject, originalParseObject, config) {\n  if (!parseObject) {\n    return Promise.resolve({});\n  }\n  return new Promise(function (resolve, reject) {\n    var trigger = getTrigger(parseObject.className, triggerType, config.applicationId);\n    if (!trigger) return resolve();\n    var request = getRequestObject(triggerType, auth, parseObject, originalParseObject, config);\n    var response = getResponseObject(request, (object) => {\n      logTriggerSuccessBeforeHook(\n        triggerType, parseObject.className, parseObject.toJSON(), object, auth);\n      resolve(object);\n    }, (error) => {\n      logTriggerErrorBeforeHook(\n        triggerType, parseObject.className, parseObject.toJSON(), auth, error);\n      reject(error);\n    });\n    // Force the current Parse app before the trigger\n    Parse.applicationId = config.applicationId;\n    Parse.javascriptKey = config.javascriptKey || '';\n    Parse.masterKey = config.masterKey;\n\n    // AfterSave and afterDelete triggers can return a promise, which if they\n    // do, needs to be resolved before this promise is resolved,\n    // so trigger execution is synced with RestWrite.execute() call.\n    // If triggers do not return a promise, they can run async code parallel\n    // to the RestWrite.execute() call.\n    var triggerPromise = trigger(request, response);\n    if(triggerType === Types.afterSave || triggerType === Types.afterDelete)\n    {\n      logTriggerAfterHook(triggerType, parseObject.className, parseObject.toJSON(), auth);\n      if(triggerPromise && typeof triggerPromise.then === \"function\") {\n        return triggerPromise.then(resolve, resolve);\n      }\n      else {\n        return resolve();\n      }\n    }\n  });\n}\n\n// Converts a REST-format object to a Parse.Object\n// data is either className or an object\nexport function inflate(data, restObject) {\n  var copy = typeof data == 'object' ? data : {className: data};\n  for (var key in restObject) {\n    copy[key] = restObject[key];\n  }\n  return Parse.Object.fromJSON(copy);\n}\n\nexport function runLiveQueryEventHandlers(data, applicationId = Parse.applicationId) {\n  if (!_triggerStore || !_triggerStore[applicationId] || !_triggerStore[applicationId].LiveQuery) { return; }\n  _triggerStore[applicationId].LiveQuery.forEach((handler) => handler(data));\n}\n"]}
|
|
844
|
+
function getRequestFileObject(triggerType, auth, fileObject, config) {
|
|
845
|
+
const request = {
|
|
846
|
+
...fileObject,
|
|
847
|
+
triggerName: triggerType,
|
|
848
|
+
master: false,
|
|
849
|
+
log: config.loggerController,
|
|
850
|
+
headers: config.headers,
|
|
851
|
+
ip: config.ip,
|
|
852
|
+
config
|
|
853
|
+
};
|
|
854
|
+
if (!auth) {
|
|
855
|
+
return request;
|
|
856
|
+
}
|
|
857
|
+
if (auth.isMaster) {
|
|
858
|
+
request['master'] = true;
|
|
859
|
+
}
|
|
860
|
+
if (auth.user) {
|
|
861
|
+
request['user'] = auth.user;
|
|
862
|
+
}
|
|
863
|
+
if (auth.installationId) {
|
|
864
|
+
request['installationId'] = auth.installationId;
|
|
865
|
+
}
|
|
866
|
+
return request;
|
|
867
|
+
}
|
|
868
|
+
async function maybeRunFileTrigger(triggerType, fileObject, config, auth) {
|
|
869
|
+
const FileClassName = getClassName(_node.default.File);
|
|
870
|
+
const fileTrigger = getTrigger(FileClassName, triggerType, config.applicationId);
|
|
871
|
+
if (typeof fileTrigger === 'function') {
|
|
872
|
+
try {
|
|
873
|
+
const request = getRequestFileObject(triggerType, auth, fileObject, config);
|
|
874
|
+
await maybeRunValidator(request, `${triggerType}.${FileClassName}`, auth);
|
|
875
|
+
if (request.skipWithMasterKey) {
|
|
876
|
+
return fileObject;
|
|
877
|
+
}
|
|
878
|
+
const result = await fileTrigger(request);
|
|
879
|
+
if (request.forceDownload) {
|
|
880
|
+
fileObject.forceDownload = true;
|
|
881
|
+
}
|
|
882
|
+
logTriggerSuccessBeforeHook(triggerType, 'Parse.File', {
|
|
883
|
+
...fileObject.file.toJSON(),
|
|
884
|
+
fileSize: fileObject.fileSize
|
|
885
|
+
}, result, auth, config.logLevels.triggerBeforeSuccess);
|
|
886
|
+
return result || fileObject;
|
|
887
|
+
} catch (error) {
|
|
888
|
+
logTriggerErrorBeforeHook(triggerType, 'Parse.File', {
|
|
889
|
+
...fileObject.file.toJSON(),
|
|
890
|
+
fileSize: fileObject.fileSize
|
|
891
|
+
}, auth, error, config.logLevels.triggerBeforeError);
|
|
892
|
+
throw error;
|
|
893
|
+
}
|
|
894
|
+
}
|
|
895
|
+
return fileObject;
|
|
896
|
+
}
|
|
897
|
+
async function maybeRunGlobalConfigTrigger(triggerType, auth, configObject, originalConfigObject, config, context) {
|
|
898
|
+
const GlobalConfigClassName = getClassName(_node.default.Config);
|
|
899
|
+
const configTrigger = getTrigger(GlobalConfigClassName, triggerType, config.applicationId);
|
|
900
|
+
if (typeof configTrigger === 'function') {
|
|
901
|
+
try {
|
|
902
|
+
const request = getRequestObject(triggerType, auth, configObject, originalConfigObject, config, context);
|
|
903
|
+
await maybeRunValidator(request, `${triggerType}.${GlobalConfigClassName}`, auth);
|
|
904
|
+
if (request.skipWithMasterKey) {
|
|
905
|
+
return configObject;
|
|
906
|
+
}
|
|
907
|
+
const result = await configTrigger(request);
|
|
908
|
+
logTriggerSuccessBeforeHook(triggerType, 'Parse.Config', configObject, result, auth, config.logLevels.triggerBeforeSuccess);
|
|
909
|
+
return result || configObject;
|
|
910
|
+
} catch (error) {
|
|
911
|
+
logTriggerErrorBeforeHook(triggerType, 'Parse.Config', configObject, auth, error, config.logLevels.triggerBeforeError);
|
|
912
|
+
throw error;
|
|
913
|
+
}
|
|
914
|
+
}
|
|
915
|
+
return configObject;
|
|
916
|
+
}
|
|
917
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_node","_interopRequireDefault","require","_logger","e","__esModule","default","Types","exports","beforeLogin","afterLogin","afterLogout","beforePasswordResetRequest","beforeSave","afterSave","beforeDelete","afterDelete","beforeFind","afterFind","beforeConnect","beforeSubscribe","afterEvent","ConnectClassName","baseStore","Validators","Object","keys","reduce","base","key","Functions","Jobs","LiveQuery","Triggers","freeze","getClassName","parseClass","className","name","replace","validateClassNameForTriggers","type","_triggerStore","Category","getStore","category","applicationId","invalidNameRegex","test","path","split","splice","Parse","store","component","add","handler","lastComponent","logger","warn","remove","get","addFunction","functionName","validationHandler","addJob","jobName","addTrigger","addConnectTrigger","addLiveQueryEventHandler","push","removeFunction","removeTrigger","_unregisterAll","forEach","appId","toJSONwithObjects","object","toJSON","stateController","CoreManager","getObjectStateController","pending","getPendingOps","_getStateIdentifier","val","_toFullJSON","getTrigger","triggerType","runTrigger","trigger","request","auth","maybeRunValidator","skipWithMasterKey","triggerExists","undefined","getFunction","getFunctionNames","functionNames","extractFunctionNames","namespace","value","getJob","getJobs","manager","getValidator","getRequestObject","parseObject","originalParseObject","config","context","isGet","triggerName","master","log","loggerController","headers","ip","original","assign","isMaster","user","installationId","getRequestQueryObject","query","count","getResponseObject","resolve","reject","success","response","objects","map","equals","_getSaveJSON","id","error","resolveError","code","Error","SCRIPT_FAILED","message","userIdForLog","logTriggerAfterHook","input","logLevel","cleanInput","truncateLogMessage","JSON","stringify","logTriggerSuccessBeforeHook","result","cleanResult","logTriggerErrorBeforeHook","maybeRunAfterFindTrigger","classNameQuery","objectsInput","Promise","length","obj","Query","parseQueryInstance","where","withJSON","processedObjectsJSON","errorData","o","logLevels","triggerBeforeSuccess","currentObject","originalClassName","tempObjectWithClassName","fromJSON","then","responseFromTrigger","results","resultsAsJSON","triggerAfter","maybeRunQueryTrigger","restWhere","restOptions","json","parseQuery","requestObject","queryResult","jsonQuery","limit","skip","include","excludeKeys","explain","order","hint","comment","readPreference","includeReadPreference","subqueryReadPreference","Array","isArray","every","err","defaultOpts","stack","theValidator","builtInTriggerValidator","catch","VALIDATION_ERROR","options","validateMasterKey","reqUser","existed","requireUser","requireAnyUserRoles","requireAllUserRoles","requireMaster","params","requiredParam","validateOptions","opt","opts","includes","join","getType","fn","match","toString","toLowerCase","fields","optionPromises","set","constant","revert","required","optional","valType","all","userRoles","requireAllRoles","promises","getUserRoles","roles","resolvedUserRoles","resolvedRequireAll","hasRole","some","requiredRole","userKeys","requireUserKeys","maybeRunTrigger","startsWith","triggerBeforeError","promise","inflate","data","restObject","copy","runLiveQueryEventHandlers","getRequestFileObject","fileObject","maybeRunFileTrigger","FileClassName","File","fileTrigger","forceDownload","file","fileSize","maybeRunGlobalConfigTrigger","configObject","originalConfigObject","GlobalConfigClassName","Config","configTrigger"],"sources":["../src/triggers.js"],"sourcesContent":["// triggers.js\nimport Parse from 'parse/node';\nimport { logger } from './logger';\n\nexport const Types = {\n  beforeLogin: 'beforeLogin',\n  afterLogin: 'afterLogin',\n  afterLogout: 'afterLogout',\n  beforePasswordResetRequest: 'beforePasswordResetRequest',\n  beforeSave: 'beforeSave',\n  afterSave: 'afterSave',\n  beforeDelete: 'beforeDelete',\n  afterDelete: 'afterDelete',\n  beforeFind: 'beforeFind',\n  afterFind: 'afterFind',\n  beforeConnect: 'beforeConnect',\n  beforeSubscribe: 'beforeSubscribe',\n  afterEvent: 'afterEvent',\n};\n\nconst ConnectClassName = '@Connect';\n\nconst baseStore = function () {\n  const Validators = Object.keys(Types).reduce(function (base, key) {\n    base[key] = {};\n    return base;\n  }, {});\n  const Functions = {};\n  const Jobs = {};\n  const LiveQuery = [];\n  const Triggers = Object.keys(Types).reduce(function (base, key) {\n    base[key] = {};\n    return base;\n  }, {});\n\n  return Object.freeze({\n    Functions,\n    Jobs,\n    Validators,\n    Triggers,\n    LiveQuery,\n  });\n};\n\nexport function getClassName(parseClass) {\n  if (parseClass && parseClass.className) {\n    return parseClass.className;\n  }\n  if (parseClass && parseClass.name) {\n    return parseClass.name.replace('Parse', '@');\n  }\n  return parseClass;\n}\n\nfunction validateClassNameForTriggers(className, type) {\n  if (type == Types.beforeSave && className === '_PushStatus') {\n    // _PushStatus uses undocumented nested key increment ops\n    // allowing beforeSave would mess up the objects big time\n    // TODO: Allow proper documented way of using nested increment ops\n    throw 'Only afterSave is allowed on _PushStatus';\n  }\n  if ((type === Types.beforeLogin || type === Types.afterLogin || type === Types.beforePasswordResetRequest) && className !== '_User') {\n    // TODO: check if upstream code will handle `Error` instance rather\n    // than this anti-pattern of throwing strings\n    throw 'Only the _User class is allowed for the beforeLogin, afterLogin, and beforePasswordResetRequest triggers';\n  }\n  if (type === Types.afterLogout && className !== '_Session') {\n    // TODO: check if upstream code will handle `Error` instance rather\n    // than this anti-pattern of throwing strings\n    throw 'Only the _Session class is allowed for the afterLogout trigger.';\n  }\n  if (className === '_Session' && type !== Types.afterLogout) {\n    // TODO: check if upstream code will handle `Error` instance rather\n    // than this anti-pattern of throwing strings\n    throw 'Only the afterLogout trigger is allowed for the _Session class.';\n  }\n  return className;\n}\n\nconst _triggerStore = {};\n\nconst Category = {\n  Functions: 'Functions',\n  Validators: 'Validators',\n  Jobs: 'Jobs',\n  Triggers: 'Triggers',\n};\n\nfunction getStore(category, name, applicationId) {\n  const invalidNameRegex = /['\"`]/;\n  if (invalidNameRegex.test(name)) {\n    // Prevent a malicious user from injecting properties into the store\n    return {};\n  }\n\n  const path = name.split('.');\n  path.splice(-1); // remove last component\n  applicationId = applicationId || Parse.applicationId;\n  _triggerStore[applicationId] = _triggerStore[applicationId] || baseStore();\n  let store = _triggerStore[applicationId][category];\n  for (const component of path) {\n    store = store[component];\n    if (!store) {\n      return {};\n    }\n  }\n  return store;\n}\n\nfunction add(category, name, handler, applicationId) {\n  const lastComponent = name.split('.').splice(-1);\n  const store = getStore(category, name, applicationId);\n  if (store[lastComponent]) {\n    logger.warn(\n      `Warning: Duplicate cloud functions exist for ${lastComponent}. Only the last one will be used and the others will be ignored.`\n    );\n  }\n  store[lastComponent] = handler;\n}\n\nfunction remove(category, name, applicationId) {\n  const lastComponent = name.split('.').splice(-1);\n  const store = getStore(category, name, applicationId);\n  delete store[lastComponent];\n}\n\nfunction get(category, name, applicationId) {\n  const lastComponent = name.split('.').splice(-1);\n  const store = getStore(category, name, applicationId);\n  return store[lastComponent];\n}\n\nexport function addFunction(functionName, handler, validationHandler, applicationId) {\n  add(Category.Functions, functionName, handler, applicationId);\n  add(Category.Validators, functionName, validationHandler, applicationId);\n}\n\nexport function addJob(jobName, handler, applicationId) {\n  add(Category.Jobs, jobName, handler, applicationId);\n}\n\nexport function addTrigger(type, className, handler, applicationId, validationHandler) {\n  validateClassNameForTriggers(className, type);\n  add(Category.Triggers, `${type}.${className}`, handler, applicationId);\n  add(Category.Validators, `${type}.${className}`, validationHandler, applicationId);\n}\n\nexport function addConnectTrigger(type, handler, applicationId, validationHandler) {\n  add(Category.Triggers, `${type}.${ConnectClassName}`, handler, applicationId);\n  add(Category.Validators, `${type}.${ConnectClassName}`, validationHandler, applicationId);\n}\n\nexport function addLiveQueryEventHandler(handler, applicationId) {\n  applicationId = applicationId || Parse.applicationId;\n  _triggerStore[applicationId] = _triggerStore[applicationId] || baseStore();\n  _triggerStore[applicationId].LiveQuery.push(handler);\n}\n\nexport function removeFunction(functionName, applicationId) {\n  remove(Category.Functions, functionName, applicationId);\n}\n\nexport function removeTrigger(type, className, applicationId) {\n  remove(Category.Triggers, `${type}.${className}`, applicationId);\n}\n\nexport function _unregisterAll() {\n  Object.keys(_triggerStore).forEach(appId => delete _triggerStore[appId]);\n}\n\nexport function toJSONwithObjects(object, className) {\n  if (!object || !object.toJSON) {\n    return {};\n  }\n  const toJSON = object.toJSON();\n  const stateController = Parse.CoreManager.getObjectStateController();\n  const [pending] = stateController.getPendingOps(object._getStateIdentifier());\n  for (const key in pending) {\n    const val = object.get(key);\n    if (!val || !val._toFullJSON) {\n      toJSON[key] = val;\n      continue;\n    }\n    toJSON[key] = val._toFullJSON();\n  }\n  // Preserve original object's className if no override className is provided\n  if (className) {\n    toJSON.className = className;\n  } else if (object.className && !toJSON.className) {\n    toJSON.className = object.className;\n  }\n  return toJSON;\n}\n\nexport function getTrigger(className, triggerType, applicationId) {\n  if (!applicationId) {\n    throw 'Missing ApplicationID';\n  }\n  return get(Category.Triggers, `${triggerType}.${className}`, applicationId);\n}\n\nexport async function runTrigger(trigger, name, request, auth) {\n  if (!trigger) {\n    return;\n  }\n  await maybeRunValidator(request, name, auth);\n  if (request.skipWithMasterKey) {\n    return;\n  }\n  return await trigger(request);\n}\n\nexport function triggerExists(className: string, type: string, applicationId: string): boolean {\n  return getTrigger(className, type, applicationId) != undefined;\n}\n\nexport function getFunction(functionName, applicationId) {\n  return get(Category.Functions, functionName, applicationId);\n}\n\nexport function getFunctionNames(applicationId) {\n  const store =\n    (_triggerStore[applicationId] && _triggerStore[applicationId][Category.Functions]) || {};\n  const functionNames = [];\n  const extractFunctionNames = (namespace, store) => {\n    Object.keys(store).forEach(name => {\n      const value = store[name];\n      if (namespace) {\n        name = `${namespace}.${name}`;\n      }\n      if (typeof value === 'function') {\n        functionNames.push(name);\n      } else {\n        extractFunctionNames(name, value);\n      }\n    });\n  };\n  extractFunctionNames(null, store);\n  return functionNames;\n}\n\nexport function getJob(jobName, applicationId) {\n  return get(Category.Jobs, jobName, applicationId);\n}\n\nexport function getJobs(applicationId) {\n  var manager = _triggerStore[applicationId];\n  if (manager && manager.Jobs) {\n    return manager.Jobs;\n  }\n  return undefined;\n}\n\nexport function getValidator(functionName, applicationId) {\n  return get(Category.Validators, functionName, applicationId);\n}\n\nexport function getRequestObject(\n  triggerType,\n  auth,\n  parseObject,\n  originalParseObject,\n  config,\n  context,\n  isGet\n) {\n  const request = {\n    triggerName: triggerType,\n    object: parseObject,\n    master: false,\n    log: config.loggerController,\n    headers: config.headers,\n    ip: config.ip,\n    config,\n  };\n\n  if (isGet !== undefined) {\n    request.isGet = !!isGet;\n  }\n\n  if (originalParseObject) {\n    request.original = originalParseObject;\n  }\n  if (\n    triggerType === Types.beforeSave ||\n    triggerType === Types.afterSave ||\n    triggerType === Types.beforeDelete ||\n    triggerType === Types.afterDelete ||\n    triggerType === Types.beforeLogin ||\n    triggerType === Types.afterLogin ||\n    triggerType === Types.beforePasswordResetRequest ||\n    triggerType === Types.afterFind\n  ) {\n    // Set a copy of the context on the request object.\n    request.context = Object.assign({}, context);\n  }\n\n  if (!auth) {\n    return request;\n  }\n  if (auth.isMaster) {\n    request['master'] = true;\n  }\n  if (auth.user) {\n    request['user'] = auth.user;\n  }\n  if (auth.installationId) {\n    request['installationId'] = auth.installationId;\n  }\n  return request;\n}\n\nexport function getRequestQueryObject(triggerType, auth, query, count, config, context, isGet) {\n  isGet = !!isGet;\n\n  var request = {\n    triggerName: triggerType,\n    query,\n    master: false,\n    count,\n    log: config.loggerController,\n    isGet,\n    headers: config.headers,\n    ip: config.ip,\n    context: context || {},\n    config,\n  };\n\n  if (!auth) {\n    return request;\n  }\n  if (auth.isMaster) {\n    request['master'] = true;\n  }\n  if (auth.user) {\n    request['user'] = auth.user;\n  }\n  if (auth.installationId) {\n    request['installationId'] = auth.installationId;\n  }\n  return request;\n}\n\n// Creates the response object, and uses the request object to pass data\n// The API will call this with REST API formatted objects, this will\n// transform them to Parse.Object instances expected by Cloud Code.\n// Any changes made to the object in a beforeSave will be included.\nexport function getResponseObject(request, resolve, reject) {\n  return {\n    success: function (response) {\n      if (request.triggerName === Types.afterFind) {\n        if (!response) {\n          response = request.objects;\n        }\n        response = response.map(object => {\n          return toJSONwithObjects(object);\n        });\n        return resolve(response);\n      }\n      // Use the JSON response\n      if (\n        response &&\n        typeof response === 'object' &&\n        !request.object.equals(response) &&\n        request.triggerName === Types.beforeSave\n      ) {\n        return resolve(response);\n      }\n      if (response && typeof response === 'object' && request.triggerName === Types.afterSave) {\n        return resolve(response);\n      }\n      if (request.triggerName === Types.afterSave) {\n        return resolve();\n      }\n      response = {};\n      if (request.triggerName === Types.beforeSave) {\n        response['object'] = request.object._getSaveJSON();\n        response['object']['objectId'] = request.object.id;\n      }\n      return resolve(response);\n    },\n    error: function (error) {\n      const e = resolveError(error, {\n        code: Parse.Error.SCRIPT_FAILED,\n        message: 'Script failed. Unknown error.',\n      });\n      reject(e);\n    },\n  };\n}\n\nfunction userIdForLog(auth) {\n  return auth && auth.user ? auth.user.id : undefined;\n}\n\nfunction logTriggerAfterHook(triggerType, className, input, auth, logLevel) {\n  if (logLevel === 'silent') {\n    return;\n  }\n  const cleanInput = logger.truncateLogMessage(JSON.stringify(input));\n  logger[logLevel](\n    `${triggerType} triggered for ${className} for user ${userIdForLog(\n      auth\n    )}:\\n  Input: ${cleanInput}`,\n    {\n      className,\n      triggerType,\n      user: userIdForLog(auth),\n    }\n  );\n}\n\nfunction logTriggerSuccessBeforeHook(triggerType, className, input, result, auth, logLevel) {\n  if (logLevel === 'silent') {\n    return;\n  }\n  const cleanInput = logger.truncateLogMessage(JSON.stringify(input));\n  const cleanResult = logger.truncateLogMessage(JSON.stringify(result));\n  logger[logLevel](\n    `${triggerType} triggered for ${className} for user ${userIdForLog(\n      auth\n    )}:\\n  Input: ${cleanInput}\\n  Result: ${cleanResult}`,\n    {\n      className,\n      triggerType,\n      user: userIdForLog(auth),\n    }\n  );\n}\n\nfunction logTriggerErrorBeforeHook(triggerType, className, input, auth, error, logLevel) {\n  if (logLevel === 'silent') {\n    return;\n  }\n  const cleanInput = logger.truncateLogMessage(JSON.stringify(input));\n  logger[logLevel](\n    `${triggerType} failed for ${className} for user ${userIdForLog(\n      auth\n    )}:\\n  Input: ${cleanInput}\\n  Error: ${JSON.stringify(error)}`,\n    {\n      className,\n      triggerType,\n      error,\n      user: userIdForLog(auth),\n    }\n  );\n}\n\nexport function maybeRunAfterFindTrigger(\n  triggerType,\n  auth,\n  classNameQuery,\n  objectsInput,\n  config,\n  query,\n  context,\n  isGet\n) {\n  return new Promise((resolve, reject) => {\n    const trigger = getTrigger(classNameQuery, triggerType, config.applicationId);\n\n    if (!trigger) {\n      if (objectsInput && objectsInput.length > 0 && objectsInput[0] instanceof Parse.Object) {\n        return resolve(objectsInput.map(obj => toJSONwithObjects(obj)));\n      }\n      return resolve(objectsInput || []);\n    }\n\n    const request = getRequestObject(triggerType, auth, null, null, config, context, isGet);\n    // Convert query parameter to Parse.Query instance\n    if (query instanceof Parse.Query) {\n      request.query = query;\n    } else if (typeof query === 'object' && query !== null) {\n      const parseQueryInstance = new Parse.Query(classNameQuery);\n      if (query.where) {\n        parseQueryInstance.withJSON(query);\n      }\n      request.query = parseQueryInstance;\n    } else {\n      request.query = new Parse.Query(classNameQuery);\n    }\n\n    const { success, error } = getResponseObject(\n      request,\n      processedObjectsJSON => {\n        resolve(processedObjectsJSON);\n      },\n      errorData => {\n        reject(errorData);\n      }\n    );\n    logTriggerSuccessBeforeHook(\n      triggerType,\n      classNameQuery,\n      'AfterFind Input (Pre-Transform)',\n      JSON.stringify(\n        objectsInput.map(o => (o instanceof Parse.Object ? o.id + ':' + o.className : o))\n      ),\n      auth,\n      config.logLevels.triggerBeforeSuccess\n    );\n\n    // Convert plain objects to Parse.Object instances for trigger\n    request.objects = objectsInput.map(currentObject => {\n      if (currentObject instanceof Parse.Object) {\n        return currentObject;\n      }\n      // Preserve the original className if it exists, otherwise use the query className\n      const originalClassName = currentObject.className || classNameQuery;\n      const tempObjectWithClassName = { ...currentObject, className: originalClassName };\n      return Parse.Object.fromJSON(tempObjectWithClassName);\n    });\n    return Promise.resolve()\n      .then(() => {\n        return maybeRunValidator(request, `${triggerType}.${classNameQuery}`, auth);\n      })\n      .then(() => {\n        if (request.skipWithMasterKey) {\n          return request.objects;\n        }\n        const responseFromTrigger = trigger(request);\n        if (responseFromTrigger && typeof responseFromTrigger.then === 'function') {\n          return responseFromTrigger.then(results => {\n            return results;\n          });\n        }\n        return responseFromTrigger;\n      })\n      .then(success, error);\n  }).then(resultsAsJSON => {\n    logTriggerAfterHook(\n      triggerType,\n      classNameQuery,\n      JSON.stringify(resultsAsJSON),\n      auth,\n      config.logLevels.triggerAfter\n    );\n    return resultsAsJSON;\n  });\n}\n\nexport function maybeRunQueryTrigger(\n  triggerType,\n  className,\n  restWhere,\n  restOptions,\n  config,\n  auth,\n  context,\n  isGet\n) {\n  const trigger = getTrigger(className, triggerType, config.applicationId);\n  if (!trigger) {\n    return Promise.resolve({\n      restWhere,\n      restOptions,\n    });\n  }\n  const json = Object.assign({}, restOptions);\n  json.where = restWhere;\n\n  const parseQuery = new Parse.Query(className);\n  parseQuery.withJSON(json);\n\n  let count = false;\n  if (restOptions) {\n    count = !!restOptions.count;\n  }\n  const requestObject = getRequestQueryObject(\n    triggerType,\n    auth,\n    parseQuery,\n    count,\n    config,\n    context,\n    isGet\n  );\n  return Promise.resolve()\n    .then(() => {\n      return maybeRunValidator(requestObject, `${triggerType}.${className}`, auth);\n    })\n    .then(() => {\n      if (requestObject.skipWithMasterKey) {\n        return requestObject.query;\n      }\n      return trigger(requestObject);\n    })\n    .then(\n      result => {\n        let queryResult = parseQuery;\n        if (result && result instanceof Parse.Query) {\n          queryResult = result;\n        }\n        const jsonQuery = queryResult.toJSON();\n        if (jsonQuery.where) {\n          restWhere = jsonQuery.where;\n        }\n        if (jsonQuery.limit) {\n          restOptions = restOptions || {};\n          restOptions.limit = jsonQuery.limit;\n        }\n        if (jsonQuery.skip) {\n          restOptions = restOptions || {};\n          restOptions.skip = jsonQuery.skip;\n        }\n        if (jsonQuery.include) {\n          restOptions = restOptions || {};\n          restOptions.include = jsonQuery.include;\n        }\n        if (jsonQuery.excludeKeys) {\n          restOptions = restOptions || {};\n          restOptions.excludeKeys = jsonQuery.excludeKeys;\n        }\n        if (jsonQuery.explain) {\n          restOptions = restOptions || {};\n          restOptions.explain = jsonQuery.explain;\n        }\n        if (jsonQuery.keys) {\n          restOptions = restOptions || {};\n          restOptions.keys = jsonQuery.keys;\n        }\n        if (jsonQuery.order) {\n          restOptions = restOptions || {};\n          restOptions.order = jsonQuery.order;\n        }\n        if (jsonQuery.hint) {\n          restOptions = restOptions || {};\n          restOptions.hint = jsonQuery.hint;\n        }\n        if (jsonQuery.comment) {\n          restOptions = restOptions || {};\n          restOptions.comment = jsonQuery.comment;\n        }\n        if (requestObject.readPreference) {\n          restOptions = restOptions || {};\n          restOptions.readPreference = requestObject.readPreference;\n        }\n        if (requestObject.includeReadPreference) {\n          restOptions = restOptions || {};\n          restOptions.includeReadPreference = requestObject.includeReadPreference;\n        }\n        if (requestObject.subqueryReadPreference) {\n          restOptions = restOptions || {};\n          restOptions.subqueryReadPreference = requestObject.subqueryReadPreference;\n        }\n        let objects = undefined;\n        if (result instanceof Parse.Object) {\n          objects = [result];\n        } else if (\n          Array.isArray(result) &&\n          (!result.length || result.every(obj => obj instanceof Parse.Object))\n        ) {\n          objects = result;\n        }\n        return {\n          restWhere,\n          restOptions,\n          objects,\n        };\n      },\n      err => {\n        const error = resolveError(err, {\n          code: Parse.Error.SCRIPT_FAILED,\n          message: 'Script failed. Unknown error.',\n        });\n        throw error;\n      }\n    );\n}\n\nexport function resolveError(message, defaultOpts) {\n  if (!defaultOpts) {\n    defaultOpts = {};\n  }\n  if (!message) {\n    return new Parse.Error(\n      defaultOpts.code || Parse.Error.SCRIPT_FAILED,\n      defaultOpts.message || 'Script failed.'\n    );\n  }\n  if (message instanceof Parse.Error) {\n    return message;\n  }\n\n  const code = defaultOpts.code || Parse.Error.SCRIPT_FAILED;\n  // If it's an error, mark it as a script failed\n  if (typeof message === 'string') {\n    return new Parse.Error(code, message);\n  }\n  const error = new Parse.Error(code, message.message || message);\n  if (message instanceof Error) {\n    error.stack = message.stack;\n  }\n  return error;\n}\nexport function maybeRunValidator(request, functionName, auth) {\n  const theValidator = getValidator(functionName, Parse.applicationId);\n  if (!theValidator) {\n    return;\n  }\n  if (typeof theValidator === 'object' && theValidator.skipWithMasterKey && request.master) {\n    request.skipWithMasterKey = true;\n  }\n  return new Promise((resolve, reject) => {\n    return Promise.resolve()\n      .then(() => {\n        return typeof theValidator === 'object'\n          ? builtInTriggerValidator(theValidator, request, auth)\n          : theValidator(request);\n      })\n      .then(() => {\n        resolve();\n      })\n      .catch(e => {\n        const error = resolveError(e, {\n          code: Parse.Error.VALIDATION_ERROR,\n          message: 'Validation failed.',\n        });\n        reject(error);\n      });\n  });\n}\nasync function builtInTriggerValidator(options, request, auth) {\n  if (request.master && !options.validateMasterKey) {\n    return;\n  }\n  let reqUser = request.user;\n  if (\n    !reqUser &&\n    request.object &&\n    request.object.className === '_User' &&\n    !request.object.existed()\n  ) {\n    reqUser = request.object;\n  }\n  if (\n    (options.requireUser || options.requireAnyUserRoles || options.requireAllUserRoles) &&\n    !reqUser\n  ) {\n    throw 'Validation failed. Please login to continue.';\n  }\n  if (options.requireMaster && !request.master) {\n    throw 'Validation failed. Master key is required to complete this request.';\n  }\n  let params = request.params || {};\n  if (request.object) {\n    params = request.object.toJSON();\n  }\n  const requiredParam = key => {\n    const value = params[key];\n    if (value == null) {\n      throw `Validation failed. Please specify data for ${key}.`;\n    }\n  };\n\n  const validateOptions = async (opt, key, val) => {\n    let opts = opt.options;\n    if (typeof opts === 'function') {\n      try {\n        const result = await opts(val);\n        if (!result && result != null) {\n          throw opt.error || `Validation failed. Invalid value for ${key}.`;\n        }\n      } catch (e) {\n        if (!e) {\n          throw opt.error || `Validation failed. Invalid value for ${key}.`;\n        }\n\n        throw opt.error || e.message || e;\n      }\n      return;\n    }\n    if (!Array.isArray(opts)) {\n      opts = [opt.options];\n    }\n\n    if (!opts.includes(val)) {\n      throw (\n        opt.error || `Validation failed. Invalid option for ${key}. Expected: ${opts.join(', ')}`\n      );\n    }\n  };\n\n  const getType = fn => {\n    const match = fn && fn.toString().match(/^\\s*function (\\w+)/);\n    return (match ? match[1] : '').toLowerCase();\n  };\n  if (Array.isArray(options.fields)) {\n    for (const key of options.fields) {\n      requiredParam(key);\n    }\n  } else {\n    const optionPromises = [];\n    for (const key in options.fields) {\n      const opt = options.fields[key];\n      let val = params[key];\n      if (typeof opt === 'string') {\n        requiredParam(opt);\n      }\n      if (typeof opt === 'object') {\n        if (opt.default != null && val == null) {\n          val = opt.default;\n          params[key] = val;\n          if (request.object) {\n            request.object.set(key, val);\n          }\n        }\n        if (opt.constant && request.object) {\n          if (request.original) {\n            request.object.revert(key);\n          } else if (opt.default != null) {\n            request.object.set(key, opt.default);\n          }\n        }\n        if (opt.required) {\n          requiredParam(key);\n        }\n        const optional = !opt.required && val === undefined;\n        if (!optional) {\n          if (opt.type) {\n            const type = getType(opt.type);\n            const valType = Array.isArray(val) ? 'array' : typeof val;\n            if (valType !== type) {\n              throw `Validation failed. Invalid type for ${key}. Expected: ${type}`;\n            }\n          }\n          if (opt.options) {\n            optionPromises.push(validateOptions(opt, key, val));\n          }\n        }\n      }\n    }\n    await Promise.all(optionPromises);\n  }\n  let userRoles = options.requireAnyUserRoles;\n  let requireAllRoles = options.requireAllUserRoles;\n  const promises = [Promise.resolve(), Promise.resolve(), Promise.resolve()];\n  if (userRoles || requireAllRoles) {\n    promises[0] = auth.getUserRoles();\n  }\n  if (typeof userRoles === 'function') {\n    promises[1] = userRoles();\n  }\n  if (typeof requireAllRoles === 'function') {\n    promises[2] = requireAllRoles();\n  }\n  const [roles, resolvedUserRoles, resolvedRequireAll] = await Promise.all(promises);\n  if (resolvedUserRoles && Array.isArray(resolvedUserRoles)) {\n    userRoles = resolvedUserRoles;\n  }\n  if (resolvedRequireAll && Array.isArray(resolvedRequireAll)) {\n    requireAllRoles = resolvedRequireAll;\n  }\n  if (userRoles) {\n    const hasRole = userRoles.some(requiredRole => roles.includes(`role:${requiredRole}`));\n    if (!hasRole) {\n      throw `Validation failed. User does not match the required roles.`;\n    }\n  }\n  if (requireAllRoles) {\n    for (const requiredRole of requireAllRoles) {\n      if (!roles.includes(`role:${requiredRole}`)) {\n        throw `Validation failed. User does not match all the required roles.`;\n      }\n    }\n  }\n  const userKeys = options.requireUserKeys || [];\n  if (Array.isArray(userKeys)) {\n    for (const key of userKeys) {\n      if (!reqUser) {\n        throw 'Please login to make this request.';\n      }\n\n      if (reqUser.get(key) == null) {\n        throw `Validation failed. Please set data for ${key} on your account.`;\n      }\n    }\n  } else if (typeof userKeys === 'object') {\n    const optionPromises = [];\n    for (const key in options.requireUserKeys) {\n      const opt = options.requireUserKeys[key];\n      if (opt.options) {\n        optionPromises.push(validateOptions(opt, key, reqUser.get(key)));\n      }\n    }\n    await Promise.all(optionPromises);\n  }\n}\n\n// To be used as part of the promise chain when saving/deleting an object\n// Will resolve successfully if no trigger is configured\n// Resolves to an object, empty or containing an object key. A beforeSave\n// trigger will set the object key to the rest format object to save.\n// originalParseObject is optional, we only need that for before/afterSave functions\nexport function maybeRunTrigger(\n  triggerType,\n  auth,\n  parseObject,\n  originalParseObject,\n  config,\n  context\n) {\n  if (!parseObject) {\n    return Promise.resolve({});\n  }\n  return new Promise(function (resolve, reject) {\n    var trigger = getTrigger(parseObject.className, triggerType, config.applicationId);\n    if (!trigger) { return resolve(); }\n    var request = getRequestObject(\n      triggerType,\n      auth,\n      parseObject,\n      originalParseObject,\n      config,\n      context\n    );\n    var { success, error } = getResponseObject(\n      request,\n      object => {\n        logTriggerSuccessBeforeHook(\n          triggerType,\n          parseObject.className,\n          parseObject.toJSON(),\n          object,\n          auth,\n          triggerType.startsWith('after')\n            ? config.logLevels.triggerAfter\n            : config.logLevels.triggerBeforeSuccess\n        );\n        if (\n          triggerType === Types.beforeSave ||\n          triggerType === Types.afterSave ||\n          triggerType === Types.beforeDelete ||\n          triggerType === Types.afterDelete\n        ) {\n          Object.assign(context, request.context);\n        }\n        resolve(object);\n      },\n      error => {\n        logTriggerErrorBeforeHook(\n          triggerType,\n          parseObject.className,\n          parseObject.toJSON(),\n          auth,\n          error,\n          config.logLevels.triggerBeforeError\n        );\n        reject(error);\n      }\n    );\n\n    // AfterSave and afterDelete triggers can return a promise, which if they\n    // do, needs to be resolved before this promise is resolved,\n    // so trigger execution is synced with RestWrite.execute() call.\n    // If triggers do not return a promise, they can run async code parallel\n    // to the RestWrite.execute() call.\n    return Promise.resolve()\n      .then(() => {\n        return maybeRunValidator(request, `${triggerType}.${parseObject.className}`, auth);\n      })\n      .then(() => {\n        if (request.skipWithMasterKey) {\n          return Promise.resolve();\n        }\n        const promise = trigger(request);\n        if (\n          triggerType === Types.afterSave ||\n          triggerType === Types.afterDelete ||\n          triggerType === Types.afterLogin\n        ) {\n          logTriggerAfterHook(\n            triggerType,\n            parseObject.className,\n            parseObject.toJSON(),\n            auth,\n            config.logLevels.triggerAfter\n          );\n        }\n        // beforeSave is expected to return null (nothing)\n        if (triggerType === Types.beforeSave) {\n          if (promise && typeof promise.then === 'function') {\n            return promise.then(response => {\n              // response.object may come from express routing before hook\n              if (response && response.object) {\n                return response;\n              }\n              return null;\n            });\n          }\n          return null;\n        }\n\n        return promise;\n      })\n      .then(success, error);\n  });\n}\n\n// Converts a REST-format object to a Parse.Object\n// data is either className or an object\nexport function inflate(data, restObject) {\n  var copy = typeof data == 'object' ? data : { className: data };\n  for (var key in restObject) {\n    copy[key] = restObject[key];\n  }\n  return Parse.Object.fromJSON(copy);\n}\n\nexport function runLiveQueryEventHandlers(data, applicationId = Parse.applicationId) {\n  if (!_triggerStore || !_triggerStore[applicationId] || !_triggerStore[applicationId].LiveQuery) {\n    return;\n  }\n  _triggerStore[applicationId].LiveQuery.forEach(handler => handler(data));\n}\n\nexport function getRequestFileObject(triggerType, auth, fileObject, config) {\n  const request = {\n    ...fileObject,\n    triggerName: triggerType,\n    master: false,\n    log: config.loggerController,\n    headers: config.headers,\n    ip: config.ip,\n    config,\n  };\n\n  if (!auth) {\n    return request;\n  }\n  if (auth.isMaster) {\n    request['master'] = true;\n  }\n  if (auth.user) {\n    request['user'] = auth.user;\n  }\n  if (auth.installationId) {\n    request['installationId'] = auth.installationId;\n  }\n  return request;\n}\n\nexport async function maybeRunFileTrigger(triggerType, fileObject, config, auth) {\n  const FileClassName = getClassName(Parse.File);\n  const fileTrigger = getTrigger(FileClassName, triggerType, config.applicationId);\n  if (typeof fileTrigger === 'function') {\n    try {\n      const request = getRequestFileObject(triggerType, auth, fileObject, config);\n      await maybeRunValidator(request, `${triggerType}.${FileClassName}`, auth);\n      if (request.skipWithMasterKey) {\n        return fileObject;\n      }\n      const result = await fileTrigger(request);\n      if (request.forceDownload) {\n        fileObject.forceDownload = true;\n      }\n      logTriggerSuccessBeforeHook(\n        triggerType,\n        'Parse.File',\n        { ...fileObject.file.toJSON(), fileSize: fileObject.fileSize },\n        result,\n        auth,\n        config.logLevels.triggerBeforeSuccess\n      );\n      return result || fileObject;\n    } catch (error) {\n      logTriggerErrorBeforeHook(\n        triggerType,\n        'Parse.File',\n        { ...fileObject.file.toJSON(), fileSize: fileObject.fileSize },\n        auth,\n        error,\n        config.logLevels.triggerBeforeError\n      );\n      throw error;\n    }\n  }\n  return fileObject;\n}\n\nexport async function maybeRunGlobalConfigTrigger(triggerType, auth, configObject, originalConfigObject, config, context) {\n  const GlobalConfigClassName = getClassName(Parse.Config);\n  const configTrigger = getTrigger(GlobalConfigClassName, triggerType, config.applicationId);\n  if (typeof configTrigger === 'function') {\n    try {\n      const request = getRequestObject(triggerType, auth, configObject, originalConfigObject, config, context);\n      await maybeRunValidator(request, `${triggerType}.${GlobalConfigClassName}`, auth);\n      if (request.skipWithMasterKey) {\n        return configObject;\n      }\n      const result = await configTrigger(request);\n      logTriggerSuccessBeforeHook(\n        triggerType,\n        'Parse.Config',\n        configObject,\n        result,\n        auth,\n        config.logLevels.triggerBeforeSuccess\n      );\n      return result || configObject;\n    } catch (error) {\n      logTriggerErrorBeforeHook(\n        triggerType,\n        'Parse.Config',\n        configObject,\n        auth,\n        error,\n        config.logLevels.triggerBeforeError\n      );\n      throw error;\n    }\n  }\n  return configObject;\n}\n"],"mappings":"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","ignoreList":[]}
|