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/rest.js
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
|
|
1
|
+
"use strict";
|
|
2
2
|
|
|
3
3
|
// This file contains helpers for running operations in REST format.
|
|
4
4
|
// The goal is that handlers that explicitly handle an express route
|
|
@@ -10,171 +10,246 @@
|
|
|
10
10
|
// things.
|
|
11
11
|
|
|
12
12
|
var Parse = require('parse/node').Parse;
|
|
13
|
-
|
|
14
13
|
var RestQuery = require('./RestQuery');
|
|
15
14
|
var RestWrite = require('./RestWrite');
|
|
16
15
|
var triggers = require('./triggers');
|
|
17
|
-
|
|
16
|
+
const {
|
|
17
|
+
enforceRoleSecurity
|
|
18
|
+
} = require('./SharedRest');
|
|
19
|
+
const {
|
|
20
|
+
createSanitizedError
|
|
21
|
+
} = require('./Error');
|
|
18
22
|
function checkTriggers(className, config, types) {
|
|
19
23
|
return types.some(triggerType => {
|
|
20
24
|
return triggers.getTrigger(className, triggers.Types[triggerType], config.applicationId);
|
|
21
25
|
});
|
|
22
26
|
}
|
|
23
|
-
|
|
24
27
|
function checkLiveQuery(className, config) {
|
|
25
28
|
return config.liveQueryController && config.liveQueryController.hasLiveQuery(className);
|
|
26
29
|
}
|
|
30
|
+
async function runFindTriggers(config, auth, className, restWhere, restOptions, clientSDK, context, options = {}) {
|
|
31
|
+
const {
|
|
32
|
+
isGet
|
|
33
|
+
} = options;
|
|
34
|
+
if (restOptions && restOptions.explain && !auth.isMaster) {
|
|
35
|
+
const allowPublicExplain = config.databaseOptions?.allowPublicExplain ?? true;
|
|
36
|
+
if (!allowPublicExplain) {
|
|
37
|
+
throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Using the explain query parameter requires the master key');
|
|
38
|
+
}
|
|
39
|
+
}
|
|
27
40
|
|
|
28
|
-
//
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
41
|
+
// Run beforeFind trigger - may modify query or return objects directly
|
|
42
|
+
const result = await triggers.maybeRunQueryTrigger(triggers.Types.beforeFind, className, restWhere, restOptions, config, auth, context, isGet);
|
|
43
|
+
restWhere = result.restWhere || restWhere;
|
|
44
|
+
restOptions = result.restOptions || restOptions;
|
|
45
|
+
|
|
46
|
+
// Short-circuit path: beforeFind returned objects directly
|
|
47
|
+
// Security risk: These objects may have been fetched with master privileges
|
|
48
|
+
if (result?.objects) {
|
|
49
|
+
const objectsFromBeforeFind = result.objects;
|
|
50
|
+
let objectsForAfterFind = objectsFromBeforeFind;
|
|
51
|
+
|
|
52
|
+
// Security check: Re-filter objects if not master to ensure ACL/CLP compliance
|
|
53
|
+
if (!auth?.isMaster && !auth?.isMaintenance) {
|
|
54
|
+
const ids = (Array.isArray(objectsFromBeforeFind) ? objectsFromBeforeFind : [objectsFromBeforeFind]).map(o => o && (o.id || o.objectId) || null).filter(Boolean);
|
|
55
|
+
|
|
56
|
+
// Objects without IDs are(normally) unsaved objects
|
|
57
|
+
// For unsaved objects, the ACL security does not apply, so no need to redo the query.
|
|
58
|
+
// For saved objects, we need to re-query to ensure proper ACL/CLP enforcement
|
|
59
|
+
if (ids.length > 0) {
|
|
60
|
+
const refilterWhere = isGet ? {
|
|
61
|
+
objectId: ids[0]
|
|
62
|
+
} : {
|
|
63
|
+
objectId: {
|
|
64
|
+
$in: ids
|
|
65
|
+
}
|
|
66
|
+
};
|
|
67
|
+
|
|
68
|
+
// Re-query with proper security: no triggers to avoid infinite loops
|
|
69
|
+
const refilterQuery = await RestQuery({
|
|
70
|
+
method: isGet ? RestQuery.Method.get : RestQuery.Method.find,
|
|
71
|
+
config,
|
|
72
|
+
auth,
|
|
73
|
+
className,
|
|
74
|
+
restWhere: refilterWhere,
|
|
75
|
+
restOptions,
|
|
76
|
+
clientSDK,
|
|
77
|
+
context,
|
|
78
|
+
runBeforeFind: false,
|
|
79
|
+
runAfterFind: false
|
|
80
|
+
});
|
|
81
|
+
const refiltered = await refilterQuery.execute();
|
|
82
|
+
objectsForAfterFind = refiltered && refiltered.results || [];
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// Run afterFind trigger on security-filtered objects
|
|
87
|
+
const afterFindProcessedObjects = await triggers.maybeRunAfterFindTrigger(triggers.Types.afterFind, auth, className, objectsForAfterFind, config, new Parse.Query(className).withJSON({
|
|
88
|
+
where: restWhere,
|
|
89
|
+
...restOptions
|
|
90
|
+
}), context, isGet);
|
|
91
|
+
return {
|
|
92
|
+
results: afterFindProcessedObjects
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
// Normal path: execute database query with modified conditions
|
|
97
|
+
const query = await RestQuery({
|
|
98
|
+
method: isGet ? RestQuery.Method.get : RestQuery.Method.find,
|
|
99
|
+
config,
|
|
100
|
+
auth,
|
|
101
|
+
className,
|
|
102
|
+
restWhere,
|
|
103
|
+
restOptions,
|
|
104
|
+
clientSDK,
|
|
105
|
+
context,
|
|
106
|
+
runBeforeFind: false
|
|
36
107
|
});
|
|
108
|
+
return query.execute();
|
|
37
109
|
}
|
|
38
110
|
|
|
111
|
+
// Returns a promise for an object with optional keys 'results' and 'count'.
|
|
112
|
+
const find = async (config, auth, className, restWhere, restOptions, clientSDK, context) => {
|
|
113
|
+
enforceRoleSecurity('find', className, auth, config);
|
|
114
|
+
return runFindTriggers(config, auth, className, restWhere, restOptions, clientSDK, context, {
|
|
115
|
+
isGet: false
|
|
116
|
+
});
|
|
117
|
+
};
|
|
118
|
+
|
|
39
119
|
// get is just like find but only queries an objectId.
|
|
40
|
-
const get = (config, auth, className, objectId, restOptions, clientSDK) => {
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
const query = new RestQuery(config, auth, className, restWhere, restOptions, clientSDK);
|
|
47
|
-
return query.execute();
|
|
120
|
+
const get = async (config, auth, className, objectId, restOptions, clientSDK, context) => {
|
|
121
|
+
enforceRoleSecurity('get', className, auth, config);
|
|
122
|
+
return runFindTriggers(config, auth, className, {
|
|
123
|
+
objectId
|
|
124
|
+
}, restOptions, clientSDK, context, {
|
|
125
|
+
isGet: true
|
|
48
126
|
});
|
|
49
127
|
};
|
|
50
128
|
|
|
51
129
|
// Returns a promise that doesn't resolve to any useful value.
|
|
52
|
-
function del(config, auth, className, objectId) {
|
|
130
|
+
function del(config, auth, className, objectId, context) {
|
|
53
131
|
if (typeof objectId !== 'string') {
|
|
54
132
|
throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad objectId');
|
|
55
133
|
}
|
|
56
|
-
|
|
57
134
|
if (className === '_User' && auth.isUnauthenticated()) {
|
|
58
135
|
throw new Parse.Error(Parse.Error.SESSION_MISSING, 'Insufficient auth to delete user');
|
|
59
136
|
}
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
return Promise.resolve().then(() => {
|
|
137
|
+
enforceRoleSecurity('delete', className, auth, config);
|
|
138
|
+
let inflatedObject;
|
|
139
|
+
let schemaController;
|
|
140
|
+
return Promise.resolve().then(async () => {
|
|
66
141
|
const hasTriggers = checkTriggers(className, config, ['beforeDelete', 'afterDelete']);
|
|
67
142
|
const hasLiveQuery = checkLiveQuery(className, config);
|
|
68
143
|
if (hasTriggers || hasLiveQuery || className == '_Session') {
|
|
69
|
-
|
|
144
|
+
const query = await RestQuery({
|
|
145
|
+
method: RestQuery.Method.get,
|
|
146
|
+
config,
|
|
147
|
+
auth,
|
|
148
|
+
className,
|
|
149
|
+
restWhere: {
|
|
150
|
+
objectId
|
|
151
|
+
}
|
|
152
|
+
});
|
|
153
|
+
return query.execute({
|
|
154
|
+
op: 'delete'
|
|
155
|
+
}).then(response => {
|
|
70
156
|
if (response && response.results && response.results.length) {
|
|
71
157
|
const firstResult = response.results[0];
|
|
72
158
|
firstResult.className = className;
|
|
73
|
-
if (className === '_Session' && !auth.isMaster) {
|
|
159
|
+
if (className === '_Session' && !auth.isMaster && !auth.isMaintenance) {
|
|
74
160
|
if (!auth.user || firstResult.user.objectId !== auth.user.id) {
|
|
75
|
-
throw
|
|
161
|
+
throw createSanitizedError(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token', config);
|
|
76
162
|
}
|
|
77
163
|
}
|
|
78
164
|
var cacheAdapter = config.cacheController;
|
|
79
165
|
cacheAdapter.user.del(firstResult.sessionToken);
|
|
80
166
|
inflatedObject = Parse.Object.fromJSON(firstResult);
|
|
81
|
-
|
|
82
|
-
config.liveQueryController.onAfterDelete(inflatedObject.className, inflatedObject);
|
|
83
|
-
return triggers.maybeRunTrigger(triggers.Types.beforeDelete, auth, inflatedObject, null, config);
|
|
167
|
+
return triggers.maybeRunTrigger(triggers.Types.beforeDelete, auth, inflatedObject, null, config, context);
|
|
84
168
|
}
|
|
85
169
|
throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found for delete.');
|
|
86
170
|
});
|
|
87
171
|
}
|
|
88
172
|
return Promise.resolve({});
|
|
89
173
|
}).then(() => {
|
|
90
|
-
if (!auth.isMaster) {
|
|
174
|
+
if (!auth.isMaster && !auth.isMaintenance) {
|
|
91
175
|
return auth.getUserRoles();
|
|
92
176
|
} else {
|
|
93
177
|
return;
|
|
94
178
|
}
|
|
95
|
-
}).then(() => {
|
|
96
|
-
|
|
97
|
-
|
|
179
|
+
}).then(() => config.database.loadSchema()).then(s => {
|
|
180
|
+
schemaController = s;
|
|
181
|
+
const options = {};
|
|
182
|
+
if (!auth.isMaster && !auth.isMaintenance) {
|
|
98
183
|
options.acl = ['*'];
|
|
99
184
|
if (auth.user) {
|
|
100
185
|
options.acl.push(auth.user.id);
|
|
101
186
|
options.acl = options.acl.concat(auth.userRoles);
|
|
102
187
|
}
|
|
103
188
|
}
|
|
104
|
-
|
|
105
189
|
return config.database.destroy(className, {
|
|
106
190
|
objectId: objectId
|
|
107
|
-
}, options);
|
|
191
|
+
}, options, schemaController);
|
|
108
192
|
}).then(() => {
|
|
109
|
-
|
|
193
|
+
// Notify LiveQuery server if possible
|
|
194
|
+
const perms = schemaController.getClassLevelPermissions(className);
|
|
195
|
+
config.liveQueryController.onAfterDelete(className, inflatedObject, null, perms);
|
|
196
|
+
return triggers.maybeRunTrigger(triggers.Types.afterDelete, auth, inflatedObject, null, config, context);
|
|
110
197
|
}).catch(error => {
|
|
111
|
-
handleSessionMissingError(error, className, auth);
|
|
198
|
+
handleSessionMissingError(error, className, auth, config);
|
|
112
199
|
});
|
|
113
200
|
}
|
|
114
201
|
|
|
115
202
|
// Returns a promise for a {response, status, location} object.
|
|
116
|
-
function create(config, auth, className, restObject, clientSDK) {
|
|
117
|
-
enforceRoleSecurity('create', className, auth);
|
|
118
|
-
var write = new RestWrite(config, auth, className, null, restObject, null, clientSDK);
|
|
203
|
+
function create(config, auth, className, restObject, clientSDK, context) {
|
|
204
|
+
enforceRoleSecurity('create', className, auth, config);
|
|
205
|
+
var write = new RestWrite(config, auth, className, null, restObject, null, clientSDK, context);
|
|
119
206
|
return write.execute();
|
|
120
207
|
}
|
|
121
208
|
|
|
122
209
|
// Returns a promise that contains the fields of the update that the
|
|
123
210
|
// REST API is supposed to return.
|
|
124
211
|
// Usually, this is just updatedAt.
|
|
125
|
-
function update(config, auth, className, restWhere, restObject, clientSDK) {
|
|
126
|
-
enforceRoleSecurity('update', className, auth);
|
|
127
|
-
|
|
128
|
-
return Promise.resolve().then(() => {
|
|
212
|
+
function update(config, auth, className, restWhere, restObject, clientSDK, context) {
|
|
213
|
+
enforceRoleSecurity('update', className, auth, config);
|
|
214
|
+
return Promise.resolve().then(async () => {
|
|
129
215
|
const hasTriggers = checkTriggers(className, config, ['beforeSave', 'afterSave']);
|
|
130
216
|
const hasLiveQuery = checkLiveQuery(className, config);
|
|
131
217
|
if (hasTriggers || hasLiveQuery) {
|
|
132
218
|
// Do not use find, as it runs the before finds
|
|
133
|
-
|
|
219
|
+
const query = await RestQuery({
|
|
220
|
+
method: RestQuery.Method.get,
|
|
221
|
+
config,
|
|
222
|
+
auth,
|
|
223
|
+
className,
|
|
224
|
+
restWhere,
|
|
225
|
+
runAfterFind: false,
|
|
226
|
+
runBeforeFind: false,
|
|
227
|
+
context
|
|
228
|
+
});
|
|
229
|
+
return query.execute({
|
|
230
|
+
op: 'update'
|
|
231
|
+
});
|
|
134
232
|
}
|
|
135
233
|
return Promise.resolve({});
|
|
136
|
-
}).then(({
|
|
234
|
+
}).then(({
|
|
235
|
+
results
|
|
236
|
+
}) => {
|
|
137
237
|
var originalRestObject;
|
|
138
238
|
if (results && results.length) {
|
|
139
239
|
originalRestObject = results[0];
|
|
140
240
|
}
|
|
141
|
-
return new RestWrite(config, auth, className, restWhere, restObject, originalRestObject, clientSDK).execute();
|
|
241
|
+
return new RestWrite(config, auth, className, restWhere, restObject, originalRestObject, clientSDK, context, 'update').execute();
|
|
142
242
|
}).catch(error => {
|
|
143
|
-
handleSessionMissingError(error, className, auth);
|
|
243
|
+
handleSessionMissingError(error, className, auth, config);
|
|
144
244
|
});
|
|
145
245
|
}
|
|
146
|
-
|
|
147
|
-
function handleSessionMissingError(error, className) {
|
|
246
|
+
function handleSessionMissingError(error, className, auth, config) {
|
|
148
247
|
// If we're trying to update a user without / with bad session token
|
|
149
|
-
if (className === '_User' && error.code === Parse.Error.OBJECT_NOT_FOUND) {
|
|
150
|
-
throw
|
|
248
|
+
if (className === '_User' && error.code === Parse.Error.OBJECT_NOT_FOUND && !auth.isMaster && !auth.isMaintenance) {
|
|
249
|
+
throw createSanitizedError(Parse.Error.SESSION_MISSING, 'Insufficient auth.', config);
|
|
151
250
|
}
|
|
152
251
|
throw error;
|
|
153
252
|
}
|
|
154
|
-
|
|
155
|
-
const classesWithMasterOnlyAccess = ['_JobStatus', '_PushStatus', '_Hooks', '_GlobalConfig', '_JobSchedule'];
|
|
156
|
-
// Disallowing access to the _Role collection except by master key
|
|
157
|
-
function enforceRoleSecurity(method, className, auth) {
|
|
158
|
-
if (className === '_Installation' && !auth.isMaster) {
|
|
159
|
-
if (method === 'delete' || method === 'find') {
|
|
160
|
-
const error = `Clients aren't allowed to perform the ${method} operation on the installation collection.`;
|
|
161
|
-
throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, error);
|
|
162
|
-
}
|
|
163
|
-
}
|
|
164
|
-
|
|
165
|
-
//all volatileClasses are masterKey only
|
|
166
|
-
if (classesWithMasterOnlyAccess.indexOf(className) >= 0 && !auth.isMaster) {
|
|
167
|
-
const error = `Clients aren't allowed to perform the ${method} operation on the ${className} collection.`;
|
|
168
|
-
throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, error);
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
// readOnly masterKey is not allowed
|
|
172
|
-
if (auth.isReadOnly && (method === 'delete' || method === 'create' || method === 'update')) {
|
|
173
|
-
const error = `read-only masterKey isn't allowed to perform the ${method} operation.`;
|
|
174
|
-
throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, error);
|
|
175
|
-
}
|
|
176
|
-
}
|
|
177
|
-
|
|
178
253
|
module.exports = {
|
|
179
254
|
create,
|
|
180
255
|
del,
|
|
@@ -182,4 +257,4 @@ module.exports = {
|
|
|
182
257
|
get,
|
|
183
258
|
update
|
|
184
259
|
};
|
|
185
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/rest.js"],"names":["Parse","require","RestQuery","RestWrite","triggers","checkTriggers","className","config","types","some","triggerType","getTrigger","Types","applicationId","checkLiveQuery","liveQueryController","hasLiveQuery","find","auth","restWhere","restOptions","clientSDK","enforceRoleSecurity","maybeRunQueryTrigger","beforeFind","then","result","query","execute","get","objectId","del","Error","INVALID_JSON","isUnauthenticated","SESSION_MISSING","inflatedObject","Promise","resolve","hasTriggers","forWrite","response","results","length","firstResult","isMaster","user","id","INVALID_SESSION_TOKEN","cacheAdapter","cacheController","sessionToken","Object","fromJSON","onAfterDelete","maybeRunTrigger","beforeDelete","OBJECT_NOT_FOUND","getUserRoles","options","acl","push","concat","userRoles","database","destroy","afterDelete","catch","error","handleSessionMissingError","create","restObject","write","update","originalRestObject","code","classesWithMasterOnlyAccess","method","OPERATION_FORBIDDEN","indexOf","isReadOnly","module","exports"],"mappings":"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","file":"rest.js","sourcesContent":["// This file contains helpers for running operations in REST format.\n// The goal is that handlers that explicitly handle an express route\n// should just be shallow wrappers around things in this file, but\n// these functions should not explicitly depend on the request\n// object.\n// This means that one of these handlers can support multiple\n// routes. That's useful for the routes that do really similar\n// things.\n\nvar Parse = require('parse/node').Parse;\n\nvar RestQuery = require('./RestQuery');\nvar RestWrite = require('./RestWrite');\nvar triggers = require('./triggers');\n\nfunction checkTriggers(className, config, types) {\n  return types.some((triggerType) => {\n    return triggers.getTrigger(className, triggers.Types[triggerType], config.applicationId);\n  });\n}\n\nfunction checkLiveQuery(className, config) {\n  return config.liveQueryController && config.liveQueryController.hasLiveQuery(className)\n}\n\n// Returns a promise for an object with optional keys 'results' and 'count'.\nfunction find(config, auth, className, restWhere, restOptions, clientSDK) {\n  enforceRoleSecurity('find', className, auth);\n  return triggers.maybeRunQueryTrigger(triggers.Types.beforeFind, className, restWhere, restOptions, config, auth).then((result) => {\n    restWhere = result.restWhere || restWhere;\n    restOptions = result.restOptions || restOptions;\n    const query = new RestQuery(config, auth, className, restWhere, restOptions, clientSDK);\n    return query.execute();\n  });\n}\n\n// get is just like find but only queries an objectId.\nconst get = (config, auth, className, objectId, restOptions, clientSDK) => {\n  var restWhere = { objectId };\n  enforceRoleSecurity('get', className, auth);\n  return triggers.maybeRunQueryTrigger(triggers.Types.beforeFind, className, restWhere, restOptions, config, auth, true).then((result) => {\n    restWhere = result.restWhere || restWhere;\n    restOptions = result.restOptions || restOptions;\n    const query = new RestQuery(config, auth, className, restWhere, restOptions, clientSDK);\n    return query.execute();\n  });\n}\n\n// Returns a promise that doesn't resolve to any useful value.\nfunction del(config, auth, className, objectId) {\n  if (typeof objectId !== 'string') {\n    throw new Parse.Error(Parse.Error.INVALID_JSON,\n      'bad objectId');\n  }\n\n  if (className === '_User' && auth.isUnauthenticated()) {\n    throw new Parse.Error(Parse.Error.SESSION_MISSING,\n      'Insufficient auth to delete user');\n  }\n\n  enforceRoleSecurity('delete', className, auth);\n\n  var inflatedObject;\n\n  return Promise.resolve().then(() => {\n    const hasTriggers = checkTriggers(className, config, ['beforeDelete', 'afterDelete']);\n    const hasLiveQuery = checkLiveQuery(className, config);\n    if (hasTriggers || hasLiveQuery || className == '_Session') {\n      return new RestQuery(config, auth, className, { objectId })\n        .forWrite()\n        .execute()\n        .then((response) => {\n          if (response && response.results && response.results.length) {\n            const firstResult = response.results[0];\n            firstResult.className = className;\n            if (className === '_Session' && !auth.isMaster) {\n              if (!auth.user || firstResult.user.objectId !== auth.user.id) {\n                throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n              }\n            }\n            var cacheAdapter = config.cacheController;\n            cacheAdapter.user.del(firstResult.sessionToken);\n            inflatedObject = Parse.Object.fromJSON(firstResult);\n            // Notify LiveQuery server if possible\n            config.liveQueryController.onAfterDelete(inflatedObject.className, inflatedObject);\n            return triggers.maybeRunTrigger(triggers.Types.beforeDelete, auth, inflatedObject, null,  config);\n          }\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND,\n            'Object not found for delete.');\n        });\n    }\n    return Promise.resolve({});\n  }).then(() => {\n    if (!auth.isMaster) {\n      return auth.getUserRoles();\n    } else {\n      return;\n    }\n  }).then(() => {\n    var options = {};\n    if (!auth.isMaster) {\n      options.acl = ['*'];\n      if (auth.user) {\n        options.acl.push(auth.user.id);\n        options.acl = options.acl.concat(auth.userRoles);\n      }\n    }\n\n    return config.database.destroy(className, {\n      objectId: objectId\n    }, options);\n  }).then(() => {\n    return triggers.maybeRunTrigger(triggers.Types.afterDelete, auth, inflatedObject, null, config);\n  }).catch((error) => {\n    handleSessionMissingError(error, className, auth);\n  });\n}\n\n// Returns a promise for a {response, status, location} object.\nfunction create(config, auth, className, restObject, clientSDK) {\n  enforceRoleSecurity('create', className, auth);\n  var write = new RestWrite(config, auth, className, null, restObject, null, clientSDK);\n  return write.execute();\n}\n\n// Returns a promise that contains the fields of the update that the\n// REST API is supposed to return.\n// Usually, this is just updatedAt.\nfunction update(config, auth, className, restWhere, restObject, clientSDK) {\n  enforceRoleSecurity('update', className, auth);\n\n  return Promise.resolve().then(() => {\n    const hasTriggers = checkTriggers(className, config, ['beforeSave', 'afterSave']);\n    const hasLiveQuery = checkLiveQuery(className, config);\n    if (hasTriggers || hasLiveQuery) {\n      // Do not use find, as it runs the before finds\n      return new RestQuery(config, auth, className, restWhere)\n        .forWrite()\n        .execute();\n    }\n    return Promise.resolve({});\n  }).then(({ results }) => {\n    var originalRestObject;\n    if (results && results.length) {\n      originalRestObject = results[0];\n    }\n    return new RestWrite(config, auth, className, restWhere, restObject, originalRestObject, clientSDK)\n      .execute();\n  }).catch((error) => {\n    handleSessionMissingError(error, className, auth);\n  });\n}\n\nfunction handleSessionMissingError(error, className) {\n  // If we're trying to update a user without / with bad session token\n  if (className === '_User'\n      && error.code === Parse.Error.OBJECT_NOT_FOUND) {\n    throw new Parse.Error(Parse.Error.SESSION_MISSING, 'Insufficient auth.');\n  }\n  throw error;\n}\n\nconst classesWithMasterOnlyAccess = ['_JobStatus', '_PushStatus', '_Hooks', '_GlobalConfig', '_JobSchedule'];\n// Disallowing access to the _Role collection except by master key\nfunction enforceRoleSecurity(method, className, auth) {\n  if (className === '_Installation' && !auth.isMaster) {\n    if (method === 'delete' || method === 'find') {\n      const error = `Clients aren't allowed to perform the ${method} operation on the installation collection.`\n      throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, error);\n    }\n  }\n\n  //all volatileClasses are masterKey only\n  if(classesWithMasterOnlyAccess.indexOf(className) >= 0 && !auth.isMaster){\n    const error = `Clients aren't allowed to perform the ${method} operation on the ${className} collection.`\n    throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, error);\n  }\n\n  // readOnly masterKey is not allowed\n  if (auth.isReadOnly && (method === 'delete' || method === 'create' || method === 'update')) {\n    const error = `read-only masterKey isn't allowed to perform the ${method} operation.`\n    throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, error);\n  }\n}\n\nmodule.exports = {\n  create,\n  del,\n  find,\n  get,\n  update\n};\n"]}
|
|
260
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","RestQuery","RestWrite","triggers","enforceRoleSecurity","createSanitizedError","checkTriggers","className","config","types","some","triggerType","getTrigger","Types","applicationId","checkLiveQuery","liveQueryController","hasLiveQuery","runFindTriggers","auth","restWhere","restOptions","clientSDK","context","options","isGet","explain","isMaster","allowPublicExplain","databaseOptions","Error","INVALID_QUERY","result","maybeRunQueryTrigger","beforeFind","objects","objectsFromBeforeFind","objectsForAfterFind","isMaintenance","ids","Array","isArray","map","o","id","objectId","filter","Boolean","length","refilterWhere","$in","refilterQuery","method","Method","get","find","runBeforeFind","runAfterFind","refiltered","execute","results","afterFindProcessedObjects","maybeRunAfterFindTrigger","afterFind","Query","withJSON","where","query","del","INVALID_JSON","isUnauthenticated","SESSION_MISSING","inflatedObject","schemaController","Promise","resolve","then","hasTriggers","op","response","firstResult","user","INVALID_SESSION_TOKEN","cacheAdapter","cacheController","sessionToken","Object","fromJSON","maybeRunTrigger","beforeDelete","OBJECT_NOT_FOUND","getUserRoles","database","loadSchema","s","acl","push","concat","userRoles","destroy","perms","getClassLevelPermissions","onAfterDelete","afterDelete","catch","error","handleSessionMissingError","create","restObject","write","update","originalRestObject","code","module","exports"],"sources":["../src/rest.js"],"sourcesContent":["// This file contains helpers for running operations in REST format.\n// The goal is that handlers that explicitly handle an express route\n// should just be shallow wrappers around things in this file, but\n// these functions should not explicitly depend on the request\n// object.\n// This means that one of these handlers can support multiple\n// routes. That's useful for the routes that do really similar\n// things.\n\nvar Parse = require('parse/node').Parse;\n\nvar RestQuery = require('./RestQuery');\nvar RestWrite = require('./RestWrite');\nvar triggers = require('./triggers');\nconst { enforceRoleSecurity } = require('./SharedRest');\nconst { createSanitizedError } = require('./Error');\n\nfunction checkTriggers(className, config, types) {\n  return types.some(triggerType => {\n    return triggers.getTrigger(className, triggers.Types[triggerType], config.applicationId);\n  });\n}\n\nfunction checkLiveQuery(className, config) {\n  return config.liveQueryController && config.liveQueryController.hasLiveQuery(className);\n}\nasync function runFindTriggers(\n  config,\n  auth,\n  className,\n  restWhere,\n  restOptions,\n  clientSDK,\n  context,\n  options = {}\n) {\n  const { isGet } = options;\n\n  if (restOptions && restOptions.explain && !auth.isMaster) {\n    const allowPublicExplain = config.databaseOptions?.allowPublicExplain ?? true;\n\n    if (!allowPublicExplain) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_QUERY,\n        'Using the explain query parameter requires the master key'\n      );\n    }\n  }\n\n  // Run beforeFind trigger - may modify query or return objects directly\n  const result = await triggers.maybeRunQueryTrigger(\n    triggers.Types.beforeFind,\n    className,\n    restWhere,\n    restOptions,\n    config,\n    auth,\n    context,\n    isGet\n  );\n\n  restWhere = result.restWhere || restWhere;\n  restOptions = result.restOptions || restOptions;\n\n  // Short-circuit path: beforeFind returned objects directly\n  // Security risk: These objects may have been fetched with master privileges\n  if (result?.objects) {\n    const objectsFromBeforeFind = result.objects;\n\n    let objectsForAfterFind = objectsFromBeforeFind;\n\n    // Security check: Re-filter objects if not master to ensure ACL/CLP compliance\n    if (!auth?.isMaster && !auth?.isMaintenance) {\n      const ids = (Array.isArray(objectsFromBeforeFind) ? objectsFromBeforeFind : [objectsFromBeforeFind])\n        .map(o => (o && (o.id || o.objectId)) || null)\n        .filter(Boolean);\n\n      // Objects without IDs are(normally) unsaved objects\n      // For unsaved objects, the ACL security does not apply, so no need to redo the query.\n      // For saved objects, we need to re-query to ensure proper ACL/CLP enforcement\n      if (ids.length > 0) {\n        const refilterWhere = isGet ? { objectId: ids[0] } : { objectId: { $in: ids } };\n\n        // Re-query with proper security: no triggers to avoid infinite loops\n        const refilterQuery = await RestQuery({\n          method: isGet ? RestQuery.Method.get : RestQuery.Method.find,\n          config,\n          auth,\n          className,\n          restWhere: refilterWhere,\n          restOptions,\n          clientSDK,\n          context,\n          runBeforeFind: false,\n          runAfterFind: false,\n        });\n\n        const refiltered = await refilterQuery.execute();\n        objectsForAfterFind = (refiltered && refiltered.results) || [];\n      }\n    }\n\n    // Run afterFind trigger on security-filtered objects\n    const afterFindProcessedObjects = await triggers.maybeRunAfterFindTrigger(\n      triggers.Types.afterFind,\n      auth,\n      className,\n      objectsForAfterFind,\n      config,\n      new Parse.Query(className).withJSON({ where: restWhere, ...restOptions }),\n      context,\n      isGet\n    );\n\n    return {\n      results: afterFindProcessedObjects,\n    };\n  }\n\n  // Normal path: execute database query with modified conditions\n  const query = await RestQuery({\n    method: isGet ? RestQuery.Method.get : RestQuery.Method.find,\n    config,\n    auth,\n    className,\n    restWhere,\n    restOptions,\n    clientSDK,\n    context,\n    runBeforeFind: false,\n  });\n\n  return query.execute();\n}\n\n// Returns a promise for an object with optional keys 'results' and 'count'.\nconst find = async (config, auth, className, restWhere, restOptions, clientSDK, context) => {\n  enforceRoleSecurity('find', className, auth, config);\n  return runFindTriggers(\n    config,\n    auth,\n    className,\n    restWhere,\n    restOptions,\n    clientSDK,\n    context,\n    { isGet: false }\n  );\n};\n\n// get is just like find but only queries an objectId.\nconst get = async (config, auth, className, objectId, restOptions, clientSDK, context) => {\n  enforceRoleSecurity('get', className, auth, config);\n  return runFindTriggers(\n    config,\n    auth,\n    className,\n    { objectId },\n    restOptions,\n    clientSDK,\n    context,\n    { isGet: true }\n  );\n};\n\n// Returns a promise that doesn't resolve to any useful value.\nfunction del(config, auth, className, objectId, context) {\n  if (typeof objectId !== 'string') {\n    throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad objectId');\n  }\n\n  if (className === '_User' && auth.isUnauthenticated()) {\n    throw new Parse.Error(Parse.Error.SESSION_MISSING, 'Insufficient auth to delete user');\n  }\n\n  enforceRoleSecurity('delete', className, auth, config);\n\n  let inflatedObject;\n  let schemaController;\n\n  return Promise.resolve()\n    .then(async () => {\n      const hasTriggers = checkTriggers(className, config, ['beforeDelete', 'afterDelete']);\n      const hasLiveQuery = checkLiveQuery(className, config);\n      if (hasTriggers || hasLiveQuery || className == '_Session') {\n        const query = await RestQuery({\n          method: RestQuery.Method.get,\n          config,\n          auth,\n          className,\n          restWhere: { objectId },\n        });\n        return query.execute({ op: 'delete' }).then(response => {\n          if (response && response.results && response.results.length) {\n            const firstResult = response.results[0];\n            firstResult.className = className;\n            if (className === '_Session' && !auth.isMaster && !auth.isMaintenance) {\n              if (!auth.user || firstResult.user.objectId !== auth.user.id) {\n                throw createSanitizedError(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token', config);\n              }\n            }\n            var cacheAdapter = config.cacheController;\n            cacheAdapter.user.del(firstResult.sessionToken);\n            inflatedObject = Parse.Object.fromJSON(firstResult);\n            return triggers.maybeRunTrigger(\n              triggers.Types.beforeDelete,\n              auth,\n              inflatedObject,\n              null,\n              config,\n              context\n            );\n          }\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found for delete.');\n        });\n      }\n      return Promise.resolve({});\n    })\n    .then(() => {\n      if (!auth.isMaster && !auth.isMaintenance) {\n        return auth.getUserRoles();\n      } else {\n        return;\n      }\n    })\n    .then(() => config.database.loadSchema())\n    .then(s => {\n      schemaController = s;\n      const options = {};\n      if (!auth.isMaster && !auth.isMaintenance) {\n        options.acl = ['*'];\n        if (auth.user) {\n          options.acl.push(auth.user.id);\n          options.acl = options.acl.concat(auth.userRoles);\n        }\n      }\n\n      return config.database.destroy(\n        className,\n        {\n          objectId: objectId,\n        },\n        options,\n        schemaController\n      );\n    })\n    .then(() => {\n      // Notify LiveQuery server if possible\n      const perms = schemaController.getClassLevelPermissions(className);\n      config.liveQueryController.onAfterDelete(className, inflatedObject, null, perms);\n      return triggers.maybeRunTrigger(\n        triggers.Types.afterDelete,\n        auth,\n        inflatedObject,\n        null,\n        config,\n        context\n      );\n    })\n    .catch(error => {\n      handleSessionMissingError(error, className, auth, config);\n    });\n}\n\n// Returns a promise for a {response, status, location} object.\nfunction create(config, auth, className, restObject, clientSDK, context) {\n  enforceRoleSecurity('create', className, auth, config);\n  var write = new RestWrite(config, auth, className, null, restObject, null, clientSDK, context);\n  return write.execute();\n}\n\n// Returns a promise that contains the fields of the update that the\n// REST API is supposed to return.\n// Usually, this is just updatedAt.\nfunction update(config, auth, className, restWhere, restObject, clientSDK, context) {\n  enforceRoleSecurity('update', className, auth, config);\n\n  return Promise.resolve()\n    .then(async () => {\n      const hasTriggers = checkTriggers(className, config, ['beforeSave', 'afterSave']);\n      const hasLiveQuery = checkLiveQuery(className, config);\n      if (hasTriggers || hasLiveQuery) {\n        // Do not use find, as it runs the before finds\n        const query = await RestQuery({\n          method: RestQuery.Method.get,\n          config,\n          auth,\n          className,\n          restWhere,\n          runAfterFind: false,\n          runBeforeFind: false,\n          context,\n        });\n        return query.execute({\n          op: 'update',\n        });\n      }\n      return Promise.resolve({});\n    })\n    .then(({ results }) => {\n      var originalRestObject;\n      if (results && results.length) {\n        originalRestObject = results[0];\n      }\n      return new RestWrite(\n        config,\n        auth,\n        className,\n        restWhere,\n        restObject,\n        originalRestObject,\n        clientSDK,\n        context,\n        'update'\n      ).execute();\n    })\n    .catch(error => {\n      handleSessionMissingError(error, className, auth, config);\n    });\n}\n\nfunction handleSessionMissingError(error, className, auth, config) {\n  // If we're trying to update a user without / with bad session token\n  if (\n    className === '_User' &&\n    error.code === Parse.Error.OBJECT_NOT_FOUND &&\n    !auth.isMaster &&\n    !auth.isMaintenance\n  ) {\n    throw createSanitizedError(Parse.Error.SESSION_MISSING, 'Insufficient auth.', config);\n  }\n  throw error;\n}\n\nmodule.exports = {\n  create,\n  del,\n  find,\n  get,\n  update,\n};\n"],"mappings":"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","ignoreList":[]}
|