parse-server 6.0.0-alpha.2 → 6.0.0-alpha.21
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +45 -17
- package/lib/AccountLockout.js +11 -26
- package/lib/Adapters/AdapterLoader.js +8 -14
- package/lib/Adapters/Analytics/AnalyticsAdapter.js +2 -8
- package/lib/Adapters/Auth/AuthAdapter.js +7 -16
- package/lib/Adapters/Auth/OAuth1Client.js +32 -57
- package/lib/Adapters/Auth/apple.js +6 -22
- package/lib/Adapters/Auth/facebook.js +7 -37
- package/lib/Adapters/Auth/gcenter.js +8 -37
- package/lib/Adapters/Auth/github.js +7 -10
- package/lib/Adapters/Auth/google.js +11 -34
- package/lib/Adapters/Auth/gpgames.js +5 -8
- package/lib/Adapters/Auth/httpsRequest.js +1 -7
- package/lib/Adapters/Auth/index.js +20 -65
- package/lib/Adapters/Auth/instagram.js +5 -9
- package/lib/Adapters/Auth/janraincapture.js +8 -12
- package/lib/Adapters/Auth/janrainengage.js +7 -11
- package/lib/Adapters/Auth/keycloak.js +5 -19
- package/lib/Adapters/Auth/ldap.js +1 -15
- package/lib/Adapters/Auth/line.js +7 -10
- package/lib/Adapters/Auth/linkedin.js +7 -12
- package/lib/Adapters/Auth/meetup.js +7 -10
- package/lib/Adapters/Auth/microsoft.js +7 -10
- package/lib/Adapters/Auth/oauth2.js +6 -18
- package/lib/Adapters/Auth/phantauth.js +8 -10
- package/lib/Adapters/Auth/qq.js +7 -13
- package/lib/Adapters/Auth/spotify.js +7 -14
- package/lib/Adapters/Auth/twitter.js +5 -15
- package/lib/Adapters/Auth/vkontakte.js +9 -15
- package/lib/Adapters/Auth/wechat.js +7 -10
- package/lib/Adapters/Auth/weibo.js +7 -11
- package/lib/Adapters/Cache/CacheAdapter.js +4 -12
- package/lib/Adapters/Cache/InMemoryCache.js +5 -19
- package/lib/Adapters/Cache/InMemoryCacheAdapter.js +1 -11
- package/lib/Adapters/Cache/LRUCache.js +1 -11
- package/lib/Adapters/Cache/NullCacheAdapter.js +1 -8
- package/lib/Adapters/Cache/RedisCacheAdapter.js +46 -87
- package/lib/Adapters/Cache/SchemaCache.js +1 -6
- package/lib/Adapters/Email/MailAdapter.js +2 -7
- package/lib/Adapters/Files/FilesAdapter.js +7 -21
- package/lib/Adapters/Files/GridFSBucketAdapter.js +6 -44
- package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
- package/lib/Adapters/Logger/LoggerAdapter.js +2 -11
- package/lib/Adapters/Logger/WinstonLogger.js +3 -30
- package/lib/Adapters/Logger/WinstonLoggerAdapter.js +5 -16
- package/lib/Adapters/MessageQueue/EventEmitterMQ.js +3 -20
- package/lib/Adapters/PubSub/EventEmitterPubSub.js +1 -16
- package/lib/Adapters/PubSub/PubSubAdapter.js +2 -9
- package/lib/Adapters/PubSub/RedisPubSub.js +13 -10
- package/lib/Adapters/Push/PushAdapter.js +2 -8
- package/lib/Adapters/Storage/Mongo/MongoCollection.js +12 -37
- package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +26 -79
- package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +78 -209
- package/lib/Adapters/Storage/Mongo/MongoTransform.js +82 -371
- package/lib/Adapters/Storage/Postgres/PostgresClient.js +1 -13
- package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +1 -20
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +119 -446
- package/lib/Adapters/Storage/Postgres/sql/index.js +4 -7
- package/lib/Adapters/Storage/StorageAdapter.js +1 -1
- package/lib/Adapters/WebSocketServer/WSAdapter.js +3 -12
- package/lib/Adapters/WebSocketServer/WSSAdapter.js +7 -12
- package/lib/Auth.js +54 -121
- package/lib/ClientSDK.js +3 -11
- package/lib/Config.js +69 -113
- package/lib/Controllers/AdaptableController.js +6 -18
- package/lib/Controllers/AnalyticsController.js +1 -9
- package/lib/Controllers/CacheController.js +3 -23
- package/lib/Controllers/DatabaseController.js +147 -345
- package/lib/Controllers/FilesController.js +5 -34
- package/lib/Controllers/HooksController.js +1 -51
- package/lib/Controllers/LiveQueryController.js +4 -23
- package/lib/Controllers/LoggerController.js +15 -54
- package/lib/Controllers/ParseGraphQLController.js +49 -104
- package/lib/Controllers/PushController.js +20 -59
- package/lib/Controllers/SchemaController.js +154 -344
- package/lib/Controllers/UserController.js +11 -72
- package/lib/Controllers/index.js +19 -68
- package/lib/Controllers/types.js +1 -1
- package/lib/Deprecator/Deprecations.js +1 -8
- package/lib/Deprecator/Deprecator.js +9 -18
- package/lib/GraphQL/ParseGraphQLSchema.js +16 -100
- package/lib/GraphQL/ParseGraphQLServer.js +2 -29
- package/lib/GraphQL/helpers/objectsMutations.js +2 -12
- package/lib/GraphQL/helpers/objectsQueries.js +18 -76
- package/lib/GraphQL/loaders/defaultGraphQLMutations.js +1 -9
- package/lib/GraphQL/loaders/defaultGraphQLQueries.js +1 -8
- package/lib/GraphQL/loaders/defaultGraphQLTypes.js +9 -115
- package/lib/GraphQL/loaders/defaultRelaySchema.js +6 -18
- package/lib/GraphQL/loaders/filesMutations.js +2 -19
- package/lib/GraphQL/loaders/functionsMutations.js +6 -17
- package/lib/GraphQL/loaders/parseClassMutations.js +6 -44
- package/lib/GraphQL/loaders/parseClassQueries.js +1 -26
- package/lib/GraphQL/loaders/parseClassTypes.js +10 -64
- package/lib/GraphQL/loaders/schemaDirectives.js +1 -17
- package/lib/GraphQL/loaders/schemaMutations.js +1 -20
- package/lib/GraphQL/loaders/schemaQueries.js +1 -14
- package/lib/GraphQL/loaders/schemaTypes.js +2 -6
- package/lib/GraphQL/loaders/usersMutations.js +6 -28
- package/lib/GraphQL/loaders/usersQueries.js +4 -26
- package/lib/GraphQL/parseGraphQLUtils.js +6 -19
- package/lib/GraphQL/transformers/className.js +1 -4
- package/lib/GraphQL/transformers/constraintType.js +1 -20
- package/lib/GraphQL/transformers/inputType.js +1 -20
- package/lib/GraphQL/transformers/mutation.js +6 -51
- package/lib/GraphQL/transformers/outputType.js +1 -20
- package/lib/GraphQL/transformers/query.js +6 -42
- package/lib/GraphQL/transformers/schemaFields.js +7 -34
- package/lib/KeyPromiseQueue.js +1 -12
- package/lib/LiveQuery/Client.js +1 -25
- package/lib/LiveQuery/Id.js +1 -7
- package/lib/LiveQuery/ParseCloudCodePublisher.js +13 -19
- package/lib/LiveQuery/ParseLiveQueryServer.js +92 -306
- package/lib/LiveQuery/ParsePubSub.js +1 -12
- package/lib/LiveQuery/ParseWebSocketServer.js +4 -26
- package/lib/LiveQuery/QueryTools.js +14 -116
- package/lib/LiveQuery/RequestSchema.js +1 -1
- package/lib/LiveQuery/SessionTokenCache.js +1 -17
- package/lib/LiveQuery/Subscription.js +4 -18
- package/lib/LiveQuery/equalObjects.js +2 -14
- package/lib/Options/Definitions.js +79 -10
- package/lib/Options/docs.js +23 -3
- package/lib/Options/index.js +4 -12
- package/lib/Options/parsers.js +1 -18
- package/lib/Page.js +1 -9
- package/lib/ParseMessageQueue.js +1 -10
- package/lib/ParseServer.js +144 -182
- package/lib/ParseServerRESTController.js +6 -33
- package/lib/PromiseRouter.js +16 -50
- package/lib/Push/PushQueue.js +3 -15
- package/lib/Push/PushWorker.js +7 -32
- package/lib/Push/utils.js +9 -38
- package/lib/RestQuery.js +105 -242
- package/lib/RestWrite.js +212 -377
- package/lib/Routers/AggregateRouter.js +14 -51
- package/lib/Routers/AnalyticsRouter.js +2 -8
- package/lib/Routers/AudiencesRouter.js +1 -15
- package/lib/Routers/ClassesRouter.js +3 -53
- package/lib/Routers/CloudCodeRouter.js +1 -19
- package/lib/Routers/FeaturesRouter.js +1 -10
- package/lib/Routers/FilesRouter.js +29 -76
- package/lib/Routers/FunctionsRouter.js +5 -28
- package/lib/Routers/GlobalConfigRouter.js +4 -18
- package/lib/Routers/GraphQLRouter.js +1 -14
- package/lib/Routers/HooksRouter.js +1 -29
- package/lib/Routers/IAPValidationRouter.js +6 -29
- package/lib/Routers/InstallationsRouter.js +2 -12
- package/lib/Routers/LogsRouter.js +4 -16
- package/lib/Routers/PagesRouter.js +69 -129
- package/lib/Routers/PublicAPIRouter.js +3 -62
- package/lib/Routers/PurgeRouter.js +1 -15
- package/lib/Routers/PushRouter.js +2 -18
- package/lib/Routers/RolesRouter.js +1 -7
- package/lib/Routers/SchemasRouter.js +4 -34
- package/lib/Routers/SecurityRouter.js +1 -12
- package/lib/Routers/SessionsRouter.js +3 -19
- package/lib/Routers/UsersRouter.js +58 -155
- package/lib/SchemaMigrations/DefinedSchemas.js +56 -115
- package/lib/SchemaMigrations/Migrations.js +2 -8
- package/lib/Security/Check.js +8 -16
- package/lib/Security/CheckGroup.js +4 -11
- package/lib/Security/CheckGroups/CheckGroupDatabase.js +8 -18
- package/lib/Security/CheckGroups/CheckGroupServerConfig.js +5 -15
- package/lib/Security/CheckGroups/CheckGroups.js +1 -4
- package/lib/Security/CheckRunner.js +22 -41
- package/lib/StatusHandler.js +12 -69
- package/lib/TestUtils.js +1 -6
- package/lib/Utils.js +27 -66
- package/lib/batch.js +17 -28
- package/lib/cache.js +1 -3
- package/lib/cli/definitions/parse-live-query-server.js +1 -3
- package/lib/cli/definitions/parse-server.js +1 -3
- package/lib/cli/parse-live-query-server.js +1 -6
- package/lib/cli/parse-server.js +11 -21
- package/lib/cli/utils/commander.js +13 -51
- package/lib/cli/utils/runner.js +1 -14
- package/lib/cloud-code/Parse.Cloud.js +71 -92
- package/lib/cryptoUtils.js +11 -19
- package/lib/defaults.js +2 -14
- package/lib/deprecated.js +1 -2
- package/lib/index.js +16 -34
- package/lib/logger.js +6 -13
- package/lib/middlewares.js +147 -151
- package/lib/password.js +6 -10
- package/lib/request.js +173 -2
- package/lib/requiredParameter.js +1 -3
- package/lib/rest.js +19 -41
- package/lib/triggers.js +54 -252
- package/lib/vendor/mongodbUrl.js +125 -305
- package/package.json +22 -19
- package/lib/cloud-code/HTTPResponse.js +0 -73
- package/lib/cloud-code/httpRequest.js +0 -192
package/lib/RestQuery.js
CHANGED
|
@@ -2,17 +2,15 @@
|
|
|
2
2
|
|
|
3
3
|
// An object that encapsulates everything we need to run a 'find'
|
|
4
4
|
// operation, encoded in the REST API format.
|
|
5
|
-
var SchemaController = require('./Controllers/SchemaController');
|
|
6
5
|
|
|
6
|
+
var SchemaController = require('./Controllers/SchemaController');
|
|
7
7
|
var Parse = require('parse/node').Parse;
|
|
8
|
-
|
|
9
8
|
const triggers = require('./triggers');
|
|
10
|
-
|
|
11
9
|
const {
|
|
12
10
|
continueWhile
|
|
13
11
|
} = require('parse/lib/node/promiseUtils');
|
|
14
|
-
|
|
15
|
-
|
|
12
|
+
const AlwaysSelectedKeys = ['objectId', 'createdAt', 'updatedAt', 'ACL'];
|
|
13
|
+
// restOptions can include:
|
|
16
14
|
// skip
|
|
17
15
|
// limit
|
|
18
16
|
// order
|
|
@@ -24,7 +22,6 @@ const AlwaysSelectedKeys = ['objectId', 'createdAt', 'updatedAt', 'ACL']; // res
|
|
|
24
22
|
// readPreference
|
|
25
23
|
// includeReadPreference
|
|
26
24
|
// subqueryReadPreference
|
|
27
|
-
|
|
28
25
|
function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, clientSDK, runAfterFind = true, context) {
|
|
29
26
|
this.config = config;
|
|
30
27
|
this.auth = auth;
|
|
@@ -36,13 +33,11 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
|
|
|
36
33
|
this.response = null;
|
|
37
34
|
this.findOptions = {};
|
|
38
35
|
this.context = context || {};
|
|
39
|
-
|
|
40
36
|
if (!this.auth.isMaster) {
|
|
41
37
|
if (this.className == '_Session') {
|
|
42
38
|
if (!this.auth.user) {
|
|
43
39
|
throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');
|
|
44
40
|
}
|
|
45
|
-
|
|
46
41
|
this.restWhere = {
|
|
47
42
|
$and: [this.restWhere, {
|
|
48
43
|
user: {
|
|
@@ -54,29 +49,29 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
|
|
|
54
49
|
};
|
|
55
50
|
}
|
|
56
51
|
}
|
|
57
|
-
|
|
58
52
|
this.doCount = false;
|
|
59
|
-
this.includeAll = false;
|
|
53
|
+
this.includeAll = false;
|
|
54
|
+
|
|
55
|
+
// The format for this.include is not the same as the format for the
|
|
60
56
|
// include option - it's the paths we should include, in order,
|
|
61
57
|
// stored as arrays, taking into account that we need to include foo
|
|
62
58
|
// before including foo.bar. Also it should dedupe.
|
|
63
59
|
// For example, passing an arg of include=foo.bar,foo.baz could lead to
|
|
64
60
|
// this.include = [['foo'], ['foo', 'baz'], ['foo', 'bar']]
|
|
65
|
-
|
|
66
61
|
this.include = [];
|
|
67
|
-
let keysForInclude = '';
|
|
68
|
-
// See issue: https://github.com/parse-community/parse-server/issues/3185
|
|
62
|
+
let keysForInclude = '';
|
|
69
63
|
|
|
64
|
+
// If we have keys, we probably want to force some includes (n-1 level)
|
|
65
|
+
// See issue: https://github.com/parse-community/parse-server/issues/3185
|
|
70
66
|
if (Object.prototype.hasOwnProperty.call(restOptions, 'keys')) {
|
|
71
67
|
keysForInclude = restOptions.keys;
|
|
72
|
-
}
|
|
73
|
-
// in order to exclude specific keys.
|
|
74
|
-
|
|
68
|
+
}
|
|
75
69
|
|
|
70
|
+
// If we have keys, we probably want to force some includes (n-1 level)
|
|
71
|
+
// in order to exclude specific keys.
|
|
76
72
|
if (Object.prototype.hasOwnProperty.call(restOptions, 'excludeKeys')) {
|
|
77
73
|
keysForInclude += ',' + restOptions.excludeKeys;
|
|
78
74
|
}
|
|
79
|
-
|
|
80
75
|
if (keysForInclude.length > 0) {
|
|
81
76
|
keysForInclude = keysForInclude.split(',').filter(key => {
|
|
82
77
|
// At least 2 components
|
|
@@ -85,9 +80,10 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
|
|
|
85
80
|
// Slice the last component (a.b.c -> a.b)
|
|
86
81
|
// Otherwise we'll include one level too much.
|
|
87
82
|
return key.slice(0, key.lastIndexOf('.'));
|
|
88
|
-
}).join(',');
|
|
89
|
-
// Dedup / sorting is handle in 'include' case.
|
|
83
|
+
}).join(',');
|
|
90
84
|
|
|
85
|
+
// Concat the possibly present include string with the one from the keys
|
|
86
|
+
// Dedup / sorting is handle in 'include' case.
|
|
91
87
|
if (keysForInclude.length > 0) {
|
|
92
88
|
if (!restOptions.include || restOptions.include.length == 0) {
|
|
93
89
|
restOptions.include = keysForInclude;
|
|
@@ -96,7 +92,6 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
|
|
|
96
92
|
}
|
|
97
93
|
}
|
|
98
94
|
}
|
|
99
|
-
|
|
100
95
|
for (var option in restOptions) {
|
|
101
96
|
switch (option) {
|
|
102
97
|
case 'keys':
|
|
@@ -105,22 +100,18 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
|
|
|
105
100
|
this.keys = Array.from(new Set(keys));
|
|
106
101
|
break;
|
|
107
102
|
}
|
|
108
|
-
|
|
109
103
|
case 'excludeKeys':
|
|
110
104
|
{
|
|
111
105
|
const exclude = restOptions.excludeKeys.split(',').filter(k => AlwaysSelectedKeys.indexOf(k) < 0);
|
|
112
106
|
this.excludeKeys = Array.from(new Set(exclude));
|
|
113
107
|
break;
|
|
114
108
|
}
|
|
115
|
-
|
|
116
109
|
case 'count':
|
|
117
110
|
this.doCount = true;
|
|
118
111
|
break;
|
|
119
|
-
|
|
120
112
|
case 'includeAll':
|
|
121
113
|
this.includeAll = true;
|
|
122
114
|
break;
|
|
123
|
-
|
|
124
115
|
case 'explain':
|
|
125
116
|
case 'hint':
|
|
126
117
|
case 'distinct':
|
|
@@ -130,12 +121,10 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
|
|
|
130
121
|
case 'readPreference':
|
|
131
122
|
this.findOptions[option] = restOptions[option];
|
|
132
123
|
break;
|
|
133
|
-
|
|
134
124
|
case 'order':
|
|
135
125
|
var fields = restOptions.order.split(',');
|
|
136
126
|
this.findOptions.sort = fields.reduce((sortMap, field) => {
|
|
137
127
|
field = field.trim();
|
|
138
|
-
|
|
139
128
|
if (field === '$score' || field === '-$score') {
|
|
140
129
|
sortMap.score = {
|
|
141
130
|
$meta: 'textScore'
|
|
@@ -145,21 +134,17 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
|
|
|
145
134
|
} else {
|
|
146
135
|
sortMap[field] = 1;
|
|
147
136
|
}
|
|
148
|
-
|
|
149
137
|
return sortMap;
|
|
150
138
|
}, {});
|
|
151
139
|
break;
|
|
152
|
-
|
|
153
140
|
case 'include':
|
|
154
141
|
{
|
|
155
142
|
const paths = restOptions.include.split(',');
|
|
156
|
-
|
|
157
143
|
if (paths.includes('*')) {
|
|
158
144
|
this.includeAll = true;
|
|
159
145
|
break;
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
|
|
146
|
+
}
|
|
147
|
+
// Load the existing includes (from keys)
|
|
163
148
|
const pathSet = paths.reduce((memo, path) => {
|
|
164
149
|
// Split each paths on . (a.b.c -> [a,b,c])
|
|
165
150
|
// reduce to create all paths
|
|
@@ -174,29 +159,27 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
|
|
|
174
159
|
}).sort((a, b) => {
|
|
175
160
|
return a.length - b.length; // Sort by number of components
|
|
176
161
|
});
|
|
162
|
+
|
|
177
163
|
break;
|
|
178
164
|
}
|
|
179
|
-
|
|
180
165
|
case 'redirectClassNameForKey':
|
|
181
166
|
this.redirectKey = restOptions.redirectClassNameForKey;
|
|
182
167
|
this.redirectClassName = null;
|
|
183
168
|
break;
|
|
184
|
-
|
|
185
169
|
case 'includeReadPreference':
|
|
186
170
|
case 'subqueryReadPreference':
|
|
187
171
|
break;
|
|
188
|
-
|
|
189
172
|
default:
|
|
190
173
|
throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad option: ' + option);
|
|
191
174
|
}
|
|
192
175
|
}
|
|
193
|
-
}
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
// A convenient method to perform all the steps of processing a query
|
|
194
179
|
// in order.
|
|
195
180
|
// Returns a promise for the response - an object with optional keys
|
|
196
181
|
// 'results' and 'count'.
|
|
197
182
|
// TODO: consolidate the replaceX functions
|
|
198
|
-
|
|
199
|
-
|
|
200
183
|
RestQuery.prototype.execute = function (executeOptions) {
|
|
201
184
|
return Promise.resolve().then(() => {
|
|
202
185
|
return this.buildRestWhere();
|
|
@@ -218,7 +201,6 @@ RestQuery.prototype.execute = function (executeOptions) {
|
|
|
218
201
|
return this.response;
|
|
219
202
|
});
|
|
220
203
|
};
|
|
221
|
-
|
|
222
204
|
RestQuery.prototype.each = function (callback) {
|
|
223
205
|
const {
|
|
224
206
|
config,
|
|
@@ -227,8 +209,8 @@ RestQuery.prototype.each = function (callback) {
|
|
|
227
209
|
restWhere,
|
|
228
210
|
restOptions,
|
|
229
211
|
clientSDK
|
|
230
|
-
} = this;
|
|
231
|
-
|
|
212
|
+
} = this;
|
|
213
|
+
// if the limit is set, use it
|
|
232
214
|
restOptions.limit = restOptions.limit || 100;
|
|
233
215
|
restOptions.order = 'objectId';
|
|
234
216
|
let finished = false;
|
|
@@ -241,7 +223,6 @@ RestQuery.prototype.each = function (callback) {
|
|
|
241
223
|
} = await query.execute();
|
|
242
224
|
results.forEach(callback);
|
|
243
225
|
finished = results.length < restOptions.limit;
|
|
244
|
-
|
|
245
226
|
if (!finished) {
|
|
246
227
|
restWhere.objectId = Object.assign({}, restWhere.objectId, {
|
|
247
228
|
$gt: results[results.length - 1].objectId
|
|
@@ -249,7 +230,6 @@ RestQuery.prototype.each = function (callback) {
|
|
|
249
230
|
}
|
|
250
231
|
});
|
|
251
232
|
};
|
|
252
|
-
|
|
253
233
|
RestQuery.prototype.buildRestWhere = function () {
|
|
254
234
|
return Promise.resolve().then(() => {
|
|
255
235
|
return this.getUserAndRoleACL();
|
|
@@ -268,16 +248,14 @@ RestQuery.prototype.buildRestWhere = function () {
|
|
|
268
248
|
}).then(() => {
|
|
269
249
|
return this.replaceEquality();
|
|
270
250
|
});
|
|
271
|
-
};
|
|
272
|
-
|
|
251
|
+
};
|
|
273
252
|
|
|
253
|
+
// Uses the Auth object to get the list of roles, adds the user id
|
|
274
254
|
RestQuery.prototype.getUserAndRoleACL = function () {
|
|
275
255
|
if (this.auth.isMaster) {
|
|
276
256
|
return Promise.resolve();
|
|
277
257
|
}
|
|
278
|
-
|
|
279
258
|
this.findOptions.acl = ['*'];
|
|
280
|
-
|
|
281
259
|
if (this.auth.user) {
|
|
282
260
|
return this.auth.getUserRoles().then(roles => {
|
|
283
261
|
this.findOptions.acl = this.findOptions.acl.concat(roles, [this.auth.user.id]);
|
|
@@ -286,23 +264,23 @@ RestQuery.prototype.getUserAndRoleACL = function () {
|
|
|
286
264
|
} else {
|
|
287
265
|
return Promise.resolve();
|
|
288
266
|
}
|
|
289
|
-
};
|
|
290
|
-
// Returns a promise.
|
|
291
|
-
|
|
267
|
+
};
|
|
292
268
|
|
|
269
|
+
// Changes the className if redirectClassNameForKey is set.
|
|
270
|
+
// Returns a promise.
|
|
293
271
|
RestQuery.prototype.redirectClassNameForKey = function () {
|
|
294
272
|
if (!this.redirectKey) {
|
|
295
273
|
return Promise.resolve();
|
|
296
|
-
}
|
|
297
|
-
|
|
274
|
+
}
|
|
298
275
|
|
|
276
|
+
// We need to change the class name based on the schema
|
|
299
277
|
return this.config.database.redirectClassNameForKey(this.className, this.redirectKey).then(newClassName => {
|
|
300
278
|
this.className = newClassName;
|
|
301
279
|
this.redirectClassName = newClassName;
|
|
302
280
|
});
|
|
303
|
-
};
|
|
304
|
-
|
|
281
|
+
};
|
|
305
282
|
|
|
283
|
+
// Validates this operation against the allowClientClassCreation config.
|
|
306
284
|
RestQuery.prototype.validateClientClassCreation = function () {
|
|
307
285
|
if (this.config.allowClientClassCreation === false && !this.auth.isMaster && SchemaController.systemClasses.indexOf(this.className) === -1) {
|
|
308
286
|
return this.config.database.loadSchema().then(schemaController => schemaController.hasClass(this.className)).then(hasClass => {
|
|
@@ -314,10 +292,8 @@ RestQuery.prototype.validateClientClassCreation = function () {
|
|
|
314
292
|
return Promise.resolve();
|
|
315
293
|
}
|
|
316
294
|
};
|
|
317
|
-
|
|
318
295
|
function transformInQuery(inQueryObject, className, results) {
|
|
319
296
|
var values = [];
|
|
320
|
-
|
|
321
297
|
for (var result of results) {
|
|
322
298
|
values.push({
|
|
323
299
|
__type: 'Pointer',
|
|
@@ -325,56 +301,47 @@ function transformInQuery(inQueryObject, className, results) {
|
|
|
325
301
|
objectId: result.objectId
|
|
326
302
|
});
|
|
327
303
|
}
|
|
328
|
-
|
|
329
304
|
delete inQueryObject['$inQuery'];
|
|
330
|
-
|
|
331
305
|
if (Array.isArray(inQueryObject['$in'])) {
|
|
332
306
|
inQueryObject['$in'] = inQueryObject['$in'].concat(values);
|
|
333
307
|
} else {
|
|
334
308
|
inQueryObject['$in'] = values;
|
|
335
309
|
}
|
|
336
|
-
}
|
|
310
|
+
}
|
|
311
|
+
|
|
312
|
+
// Replaces a $inQuery clause by running the subquery, if there is an
|
|
337
313
|
// $inQuery clause.
|
|
338
314
|
// The $inQuery clause turns into an $in with values that are just
|
|
339
315
|
// pointers to the objects returned in the subquery.
|
|
340
|
-
|
|
341
|
-
|
|
342
316
|
RestQuery.prototype.replaceInQuery = function () {
|
|
343
317
|
var inQueryObject = findObjectWithKey(this.restWhere, '$inQuery');
|
|
344
|
-
|
|
345
318
|
if (!inQueryObject) {
|
|
346
319
|
return;
|
|
347
|
-
}
|
|
348
|
-
|
|
320
|
+
}
|
|
349
321
|
|
|
322
|
+
// The inQuery value must have precisely two keys - where and className
|
|
350
323
|
var inQueryValue = inQueryObject['$inQuery'];
|
|
351
|
-
|
|
352
324
|
if (!inQueryValue.where || !inQueryValue.className) {
|
|
353
325
|
throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $inQuery');
|
|
354
326
|
}
|
|
355
|
-
|
|
356
327
|
const additionalOptions = {
|
|
357
328
|
redirectClassNameForKey: inQueryValue.redirectClassNameForKey
|
|
358
329
|
};
|
|
359
|
-
|
|
360
330
|
if (this.restOptions.subqueryReadPreference) {
|
|
361
331
|
additionalOptions.readPreference = this.restOptions.subqueryReadPreference;
|
|
362
332
|
additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;
|
|
363
333
|
} else if (this.restOptions.readPreference) {
|
|
364
334
|
additionalOptions.readPreference = this.restOptions.readPreference;
|
|
365
335
|
}
|
|
366
|
-
|
|
367
336
|
var subquery = new RestQuery(this.config, this.auth, inQueryValue.className, inQueryValue.where, additionalOptions);
|
|
368
337
|
return subquery.execute().then(response => {
|
|
369
|
-
transformInQuery(inQueryObject, subquery.className, response.results);
|
|
370
|
-
|
|
338
|
+
transformInQuery(inQueryObject, subquery.className, response.results);
|
|
339
|
+
// Recurse to repeat
|
|
371
340
|
return this.replaceInQuery();
|
|
372
341
|
});
|
|
373
342
|
};
|
|
374
|
-
|
|
375
343
|
function transformNotInQuery(notInQueryObject, className, results) {
|
|
376
344
|
var values = [];
|
|
377
|
-
|
|
378
345
|
for (var result of results) {
|
|
379
346
|
values.push({
|
|
380
347
|
__type: 'Pointer',
|
|
@@ -382,195 +349,165 @@ function transformNotInQuery(notInQueryObject, className, results) {
|
|
|
382
349
|
objectId: result.objectId
|
|
383
350
|
});
|
|
384
351
|
}
|
|
385
|
-
|
|
386
352
|
delete notInQueryObject['$notInQuery'];
|
|
387
|
-
|
|
388
353
|
if (Array.isArray(notInQueryObject['$nin'])) {
|
|
389
354
|
notInQueryObject['$nin'] = notInQueryObject['$nin'].concat(values);
|
|
390
355
|
} else {
|
|
391
356
|
notInQueryObject['$nin'] = values;
|
|
392
357
|
}
|
|
393
|
-
}
|
|
358
|
+
}
|
|
359
|
+
|
|
360
|
+
// Replaces a $notInQuery clause by running the subquery, if there is an
|
|
394
361
|
// $notInQuery clause.
|
|
395
362
|
// The $notInQuery clause turns into a $nin with values that are just
|
|
396
363
|
// pointers to the objects returned in the subquery.
|
|
397
|
-
|
|
398
|
-
|
|
399
364
|
RestQuery.prototype.replaceNotInQuery = function () {
|
|
400
365
|
var notInQueryObject = findObjectWithKey(this.restWhere, '$notInQuery');
|
|
401
|
-
|
|
402
366
|
if (!notInQueryObject) {
|
|
403
367
|
return;
|
|
404
|
-
}
|
|
405
|
-
|
|
368
|
+
}
|
|
406
369
|
|
|
370
|
+
// The notInQuery value must have precisely two keys - where and className
|
|
407
371
|
var notInQueryValue = notInQueryObject['$notInQuery'];
|
|
408
|
-
|
|
409
372
|
if (!notInQueryValue.where || !notInQueryValue.className) {
|
|
410
373
|
throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $notInQuery');
|
|
411
374
|
}
|
|
412
|
-
|
|
413
375
|
const additionalOptions = {
|
|
414
376
|
redirectClassNameForKey: notInQueryValue.redirectClassNameForKey
|
|
415
377
|
};
|
|
416
|
-
|
|
417
378
|
if (this.restOptions.subqueryReadPreference) {
|
|
418
379
|
additionalOptions.readPreference = this.restOptions.subqueryReadPreference;
|
|
419
380
|
additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;
|
|
420
381
|
} else if (this.restOptions.readPreference) {
|
|
421
382
|
additionalOptions.readPreference = this.restOptions.readPreference;
|
|
422
383
|
}
|
|
423
|
-
|
|
424
384
|
var subquery = new RestQuery(this.config, this.auth, notInQueryValue.className, notInQueryValue.where, additionalOptions);
|
|
425
385
|
return subquery.execute().then(response => {
|
|
426
|
-
transformNotInQuery(notInQueryObject, subquery.className, response.results);
|
|
427
|
-
|
|
386
|
+
transformNotInQuery(notInQueryObject, subquery.className, response.results);
|
|
387
|
+
// Recurse to repeat
|
|
428
388
|
return this.replaceNotInQuery();
|
|
429
389
|
});
|
|
430
|
-
};
|
|
431
|
-
|
|
390
|
+
};
|
|
432
391
|
|
|
392
|
+
// Used to get the deepest object from json using dot notation.
|
|
433
393
|
const getDeepestObjectFromKey = (json, key, idx, src) => {
|
|
434
394
|
if (key in json) {
|
|
435
395
|
return json[key];
|
|
436
396
|
}
|
|
437
|
-
|
|
438
397
|
src.splice(1); // Exit Early
|
|
439
398
|
};
|
|
440
399
|
|
|
441
400
|
const transformSelect = (selectObject, key, objects) => {
|
|
442
401
|
var values = [];
|
|
443
|
-
|
|
444
402
|
for (var result of objects) {
|
|
445
403
|
values.push(key.split('.').reduce(getDeepestObjectFromKey, result));
|
|
446
404
|
}
|
|
447
|
-
|
|
448
405
|
delete selectObject['$select'];
|
|
449
|
-
|
|
450
406
|
if (Array.isArray(selectObject['$in'])) {
|
|
451
407
|
selectObject['$in'] = selectObject['$in'].concat(values);
|
|
452
408
|
} else {
|
|
453
409
|
selectObject['$in'] = values;
|
|
454
410
|
}
|
|
455
|
-
};
|
|
411
|
+
};
|
|
412
|
+
|
|
413
|
+
// Replaces a $select clause by running the subquery, if there is a
|
|
456
414
|
// $select clause.
|
|
457
415
|
// The $select clause turns into an $in with values selected out of
|
|
458
416
|
// the subquery.
|
|
459
417
|
// Returns a possible-promise.
|
|
460
|
-
|
|
461
|
-
|
|
462
418
|
RestQuery.prototype.replaceSelect = function () {
|
|
463
419
|
var selectObject = findObjectWithKey(this.restWhere, '$select');
|
|
464
|
-
|
|
465
420
|
if (!selectObject) {
|
|
466
421
|
return;
|
|
467
|
-
}
|
|
468
|
-
|
|
469
|
-
|
|
470
|
-
var selectValue = selectObject['$select']; // iOS SDK don't send where if not set, let it pass
|
|
422
|
+
}
|
|
471
423
|
|
|
424
|
+
// The select value must have precisely two keys - query and key
|
|
425
|
+
var selectValue = selectObject['$select'];
|
|
426
|
+
// iOS SDK don't send where if not set, let it pass
|
|
472
427
|
if (!selectValue.query || !selectValue.key || typeof selectValue.query !== 'object' || !selectValue.query.className || Object.keys(selectValue).length !== 2) {
|
|
473
428
|
throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $select');
|
|
474
429
|
}
|
|
475
|
-
|
|
476
430
|
const additionalOptions = {
|
|
477
431
|
redirectClassNameForKey: selectValue.query.redirectClassNameForKey
|
|
478
432
|
};
|
|
479
|
-
|
|
480
433
|
if (this.restOptions.subqueryReadPreference) {
|
|
481
434
|
additionalOptions.readPreference = this.restOptions.subqueryReadPreference;
|
|
482
435
|
additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;
|
|
483
436
|
} else if (this.restOptions.readPreference) {
|
|
484
437
|
additionalOptions.readPreference = this.restOptions.readPreference;
|
|
485
438
|
}
|
|
486
|
-
|
|
487
439
|
var subquery = new RestQuery(this.config, this.auth, selectValue.query.className, selectValue.query.where, additionalOptions);
|
|
488
440
|
return subquery.execute().then(response => {
|
|
489
|
-
transformSelect(selectObject, selectValue.key, response.results);
|
|
490
|
-
|
|
441
|
+
transformSelect(selectObject, selectValue.key, response.results);
|
|
442
|
+
// Keep replacing $select clauses
|
|
491
443
|
return this.replaceSelect();
|
|
492
444
|
});
|
|
493
445
|
};
|
|
494
|
-
|
|
495
446
|
const transformDontSelect = (dontSelectObject, key, objects) => {
|
|
496
447
|
var values = [];
|
|
497
|
-
|
|
498
448
|
for (var result of objects) {
|
|
499
449
|
values.push(key.split('.').reduce(getDeepestObjectFromKey, result));
|
|
500
450
|
}
|
|
501
|
-
|
|
502
451
|
delete dontSelectObject['$dontSelect'];
|
|
503
|
-
|
|
504
452
|
if (Array.isArray(dontSelectObject['$nin'])) {
|
|
505
453
|
dontSelectObject['$nin'] = dontSelectObject['$nin'].concat(values);
|
|
506
454
|
} else {
|
|
507
455
|
dontSelectObject['$nin'] = values;
|
|
508
456
|
}
|
|
509
|
-
};
|
|
457
|
+
};
|
|
458
|
+
|
|
459
|
+
// Replaces a $dontSelect clause by running the subquery, if there is a
|
|
510
460
|
// $dontSelect clause.
|
|
511
461
|
// The $dontSelect clause turns into an $nin with values selected out of
|
|
512
462
|
// the subquery.
|
|
513
463
|
// Returns a possible-promise.
|
|
514
|
-
|
|
515
|
-
|
|
516
464
|
RestQuery.prototype.replaceDontSelect = function () {
|
|
517
465
|
var dontSelectObject = findObjectWithKey(this.restWhere, '$dontSelect');
|
|
518
|
-
|
|
519
466
|
if (!dontSelectObject) {
|
|
520
467
|
return;
|
|
521
|
-
}
|
|
522
|
-
|
|
468
|
+
}
|
|
523
469
|
|
|
470
|
+
// The dontSelect value must have precisely two keys - query and key
|
|
524
471
|
var dontSelectValue = dontSelectObject['$dontSelect'];
|
|
525
|
-
|
|
526
472
|
if (!dontSelectValue.query || !dontSelectValue.key || typeof dontSelectValue.query !== 'object' || !dontSelectValue.query.className || Object.keys(dontSelectValue).length !== 2) {
|
|
527
473
|
throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $dontSelect');
|
|
528
474
|
}
|
|
529
|
-
|
|
530
475
|
const additionalOptions = {
|
|
531
476
|
redirectClassNameForKey: dontSelectValue.query.redirectClassNameForKey
|
|
532
477
|
};
|
|
533
|
-
|
|
534
478
|
if (this.restOptions.subqueryReadPreference) {
|
|
535
479
|
additionalOptions.readPreference = this.restOptions.subqueryReadPreference;
|
|
536
480
|
additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;
|
|
537
481
|
} else if (this.restOptions.readPreference) {
|
|
538
482
|
additionalOptions.readPreference = this.restOptions.readPreference;
|
|
539
483
|
}
|
|
540
|
-
|
|
541
484
|
var subquery = new RestQuery(this.config, this.auth, dontSelectValue.query.className, dontSelectValue.query.where, additionalOptions);
|
|
542
485
|
return subquery.execute().then(response => {
|
|
543
|
-
transformDontSelect(dontSelectObject, dontSelectValue.key, response.results);
|
|
544
|
-
|
|
486
|
+
transformDontSelect(dontSelectObject, dontSelectValue.key, response.results);
|
|
487
|
+
// Keep replacing $dontSelect clauses
|
|
545
488
|
return this.replaceDontSelect();
|
|
546
489
|
});
|
|
547
490
|
};
|
|
548
|
-
|
|
549
491
|
RestQuery.prototype.cleanResultAuthData = function (result) {
|
|
550
492
|
delete result.password;
|
|
551
|
-
|
|
552
493
|
if (result.authData) {
|
|
553
494
|
Object.keys(result.authData).forEach(provider => {
|
|
554
495
|
if (result.authData[provider] === null) {
|
|
555
496
|
delete result.authData[provider];
|
|
556
497
|
}
|
|
557
498
|
});
|
|
558
|
-
|
|
559
499
|
if (Object.keys(result.authData).length == 0) {
|
|
560
500
|
delete result.authData;
|
|
561
501
|
}
|
|
562
502
|
}
|
|
563
503
|
};
|
|
564
|
-
|
|
565
504
|
const replaceEqualityConstraint = constraint => {
|
|
566
505
|
if (typeof constraint !== 'object') {
|
|
567
506
|
return constraint;
|
|
568
507
|
}
|
|
569
|
-
|
|
570
508
|
const equalToObject = {};
|
|
571
509
|
let hasDirectConstraint = false;
|
|
572
510
|
let hasOperatorConstraint = false;
|
|
573
|
-
|
|
574
511
|
for (const key in constraint) {
|
|
575
512
|
if (key.indexOf('$') !== 0) {
|
|
576
513
|
hasDirectConstraint = true;
|
|
@@ -579,29 +516,25 @@ const replaceEqualityConstraint = constraint => {
|
|
|
579
516
|
hasOperatorConstraint = true;
|
|
580
517
|
}
|
|
581
518
|
}
|
|
582
|
-
|
|
583
519
|
if (hasDirectConstraint && hasOperatorConstraint) {
|
|
584
520
|
constraint['$eq'] = equalToObject;
|
|
585
521
|
Object.keys(equalToObject).forEach(key => {
|
|
586
522
|
delete constraint[key];
|
|
587
523
|
});
|
|
588
524
|
}
|
|
589
|
-
|
|
590
525
|
return constraint;
|
|
591
526
|
};
|
|
592
|
-
|
|
593
527
|
RestQuery.prototype.replaceEquality = function () {
|
|
594
528
|
if (typeof this.restWhere !== 'object') {
|
|
595
529
|
return;
|
|
596
530
|
}
|
|
597
|
-
|
|
598
531
|
for (const key in this.restWhere) {
|
|
599
532
|
this.restWhere[key] = replaceEqualityConstraint(this.restWhere[key]);
|
|
600
533
|
}
|
|
601
|
-
};
|
|
602
|
-
// Populates this.response with an object that only has 'results'.
|
|
603
|
-
|
|
534
|
+
};
|
|
604
535
|
|
|
536
|
+
// Returns a promise for whether it was successful.
|
|
537
|
+
// Populates this.response with an object that only has 'results'.
|
|
605
538
|
RestQuery.prototype.runFind = function (options = {}) {
|
|
606
539
|
if (this.findOptions.limit === 0) {
|
|
607
540
|
this.response = {
|
|
@@ -609,47 +542,39 @@ RestQuery.prototype.runFind = function (options = {}) {
|
|
|
609
542
|
};
|
|
610
543
|
return Promise.resolve();
|
|
611
544
|
}
|
|
612
|
-
|
|
613
545
|
const findOptions = Object.assign({}, this.findOptions);
|
|
614
|
-
|
|
615
546
|
if (this.keys) {
|
|
616
547
|
findOptions.keys = this.keys.map(key => {
|
|
617
548
|
return key.split('.')[0];
|
|
618
549
|
});
|
|
619
550
|
}
|
|
620
|
-
|
|
621
551
|
if (options.op) {
|
|
622
552
|
findOptions.op = options.op;
|
|
623
553
|
}
|
|
624
|
-
|
|
625
554
|
return this.config.database.find(this.className, this.restWhere, findOptions, this.auth).then(results => {
|
|
626
555
|
if (this.className === '_User' && !findOptions.explain) {
|
|
627
556
|
for (var result of results) {
|
|
628
557
|
this.cleanResultAuthData(result);
|
|
629
558
|
}
|
|
630
559
|
}
|
|
631
|
-
|
|
632
560
|
this.config.filesController.expandFilesInObject(this.config, results);
|
|
633
|
-
|
|
634
561
|
if (this.redirectClassName) {
|
|
635
562
|
for (var r of results) {
|
|
636
563
|
r.className = this.redirectClassName;
|
|
637
564
|
}
|
|
638
565
|
}
|
|
639
|
-
|
|
640
566
|
this.response = {
|
|
641
567
|
results: results
|
|
642
568
|
};
|
|
643
569
|
});
|
|
644
|
-
};
|
|
645
|
-
// Populates this.response.count with the count
|
|
646
|
-
|
|
570
|
+
};
|
|
647
571
|
|
|
572
|
+
// Returns a promise for whether it was successful.
|
|
573
|
+
// Populates this.response.count with the count
|
|
648
574
|
RestQuery.prototype.runCount = function () {
|
|
649
575
|
if (!this.doCount) {
|
|
650
576
|
return;
|
|
651
577
|
}
|
|
652
|
-
|
|
653
578
|
this.findOptions.count = true;
|
|
654
579
|
delete this.findOptions.skip;
|
|
655
580
|
delete this.findOptions.limit;
|
|
@@ -657,73 +582,63 @@ RestQuery.prototype.runCount = function () {
|
|
|
657
582
|
this.response.count = c;
|
|
658
583
|
});
|
|
659
584
|
};
|
|
660
|
-
|
|
661
585
|
RestQuery.prototype.denyProtectedFields = async function () {
|
|
662
586
|
if (this.auth.isMaster) {
|
|
663
587
|
return;
|
|
664
588
|
}
|
|
665
|
-
|
|
666
589
|
const schemaController = await this.config.database.loadSchema();
|
|
667
590
|
const protectedFields = this.config.database.addProtectedFields(schemaController, this.className, this.restWhere, this.findOptions.acl, this.auth, this.findOptions) || [];
|
|
668
|
-
|
|
669
591
|
for (const key of protectedFields) {
|
|
670
592
|
if (this.restWhere[key]) {
|
|
671
593
|
throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, `This user is not allowed to query ${key} on class ${this.className}`);
|
|
672
594
|
}
|
|
673
595
|
}
|
|
674
|
-
};
|
|
675
|
-
|
|
596
|
+
};
|
|
676
597
|
|
|
598
|
+
// Augments this.response with all pointers on an object
|
|
677
599
|
RestQuery.prototype.handleIncludeAll = function () {
|
|
678
600
|
if (!this.includeAll) {
|
|
679
601
|
return;
|
|
680
602
|
}
|
|
681
|
-
|
|
682
603
|
return this.config.database.loadSchema().then(schemaController => schemaController.getOneSchema(this.className)).then(schema => {
|
|
683
604
|
const includeFields = [];
|
|
684
605
|
const keyFields = [];
|
|
685
|
-
|
|
686
606
|
for (const field in schema.fields) {
|
|
687
607
|
if (schema.fields[field].type && schema.fields[field].type === 'Pointer' || schema.fields[field].type && schema.fields[field].type === 'Array') {
|
|
688
608
|
includeFields.push([field]);
|
|
689
609
|
keyFields.push(field);
|
|
690
610
|
}
|
|
691
|
-
}
|
|
692
|
-
|
|
693
|
-
|
|
694
|
-
|
|
695
|
-
|
|
611
|
+
}
|
|
612
|
+
// Add fields to include, keys, remove dups
|
|
613
|
+
this.include = [...new Set([...this.include, ...includeFields])];
|
|
614
|
+
// if this.keys not set, then all keys are already included
|
|
696
615
|
if (this.keys) {
|
|
697
616
|
this.keys = [...new Set([...this.keys, ...keyFields])];
|
|
698
617
|
}
|
|
699
618
|
});
|
|
700
|
-
};
|
|
701
|
-
|
|
619
|
+
};
|
|
702
620
|
|
|
621
|
+
// Updates property `this.keys` to contain all keys but the ones unselected.
|
|
703
622
|
RestQuery.prototype.handleExcludeKeys = function () {
|
|
704
623
|
if (!this.excludeKeys) {
|
|
705
624
|
return;
|
|
706
625
|
}
|
|
707
|
-
|
|
708
626
|
if (this.keys) {
|
|
709
627
|
this.keys = this.keys.filter(k => !this.excludeKeys.includes(k));
|
|
710
628
|
return;
|
|
711
629
|
}
|
|
712
|
-
|
|
713
630
|
return this.config.database.loadSchema().then(schemaController => schemaController.getOneSchema(this.className)).then(schema => {
|
|
714
631
|
const fields = Object.keys(schema.fields);
|
|
715
632
|
this.keys = fields.filter(k => !this.excludeKeys.includes(k));
|
|
716
633
|
});
|
|
717
|
-
};
|
|
718
|
-
|
|
634
|
+
};
|
|
719
635
|
|
|
636
|
+
// Augments this.response with data at the paths provided in this.include.
|
|
720
637
|
RestQuery.prototype.handleInclude = function () {
|
|
721
638
|
if (this.include.length == 0) {
|
|
722
639
|
return;
|
|
723
640
|
}
|
|
724
|
-
|
|
725
641
|
var pathResponse = includePath(this.config, this.auth, this.response, this.include[0], this.restOptions);
|
|
726
|
-
|
|
727
642
|
if (pathResponse.then) {
|
|
728
643
|
return pathResponse.then(newResponse => {
|
|
729
644
|
this.response = newResponse;
|
|
@@ -734,37 +649,31 @@ RestQuery.prototype.handleInclude = function () {
|
|
|
734
649
|
this.include = this.include.slice(1);
|
|
735
650
|
return this.handleInclude();
|
|
736
651
|
}
|
|
737
|
-
|
|
738
652
|
return pathResponse;
|
|
739
|
-
};
|
|
740
|
-
|
|
653
|
+
};
|
|
741
654
|
|
|
655
|
+
//Returns a promise of a processed set of results
|
|
742
656
|
RestQuery.prototype.runAfterFindTrigger = function () {
|
|
743
657
|
if (!this.response) {
|
|
744
658
|
return;
|
|
745
659
|
}
|
|
746
|
-
|
|
747
660
|
if (!this.runAfterFind) {
|
|
748
661
|
return;
|
|
749
|
-
}
|
|
750
|
-
|
|
751
|
-
|
|
662
|
+
}
|
|
663
|
+
// Avoid doing any setup for triggers if there is no 'afterFind' trigger for this class.
|
|
752
664
|
const hasAfterFindHook = triggers.triggerExists(this.className, triggers.Types.afterFind, this.config.applicationId);
|
|
753
|
-
|
|
754
665
|
if (!hasAfterFindHook) {
|
|
755
666
|
return Promise.resolve();
|
|
756
|
-
}
|
|
757
|
-
|
|
758
|
-
|
|
667
|
+
}
|
|
668
|
+
// Skip Aggregate and Distinct Queries
|
|
759
669
|
if (this.findOptions.pipeline || this.findOptions.distinct) {
|
|
760
670
|
return Promise.resolve();
|
|
761
671
|
}
|
|
762
|
-
|
|
763
672
|
const json = Object.assign({}, this.restOptions);
|
|
764
673
|
json.where = this.restWhere;
|
|
765
674
|
const parseQuery = new Parse.Query(this.className);
|
|
766
|
-
parseQuery.withJSON(json);
|
|
767
|
-
|
|
675
|
+
parseQuery.withJSON(json);
|
|
676
|
+
// Run afterFind trigger and set the new results
|
|
768
677
|
return triggers.maybeRunAfterFindTrigger(triggers.Types.afterFind, this.auth, this.className, this.response.results, this.config, parseQuery, this.context).then(results => {
|
|
769
678
|
// Ensure we properly set the className back
|
|
770
679
|
if (this.redirectClassName) {
|
|
@@ -772,7 +681,6 @@ RestQuery.prototype.runAfterFindTrigger = function () {
|
|
|
772
681
|
if (object instanceof Parse.Object) {
|
|
773
682
|
object = object.toJSON();
|
|
774
683
|
}
|
|
775
|
-
|
|
776
684
|
object.className = this.redirectClassName;
|
|
777
685
|
return object;
|
|
778
686
|
});
|
|
@@ -780,94 +688,76 @@ RestQuery.prototype.runAfterFindTrigger = function () {
|
|
|
780
688
|
this.response.results = results;
|
|
781
689
|
}
|
|
782
690
|
});
|
|
783
|
-
};
|
|
691
|
+
};
|
|
692
|
+
|
|
693
|
+
// Adds included values to the response.
|
|
784
694
|
// Path is a list of field names.
|
|
785
695
|
// Returns a promise for an augmented response.
|
|
786
|
-
|
|
787
|
-
|
|
788
696
|
function includePath(config, auth, response, path, restOptions = {}) {
|
|
789
697
|
var pointers = findPointers(response.results, path);
|
|
790
|
-
|
|
791
698
|
if (pointers.length == 0) {
|
|
792
699
|
return response;
|
|
793
700
|
}
|
|
794
|
-
|
|
795
701
|
const pointersHash = {};
|
|
796
|
-
|
|
797
702
|
for (var pointer of pointers) {
|
|
798
703
|
if (!pointer) {
|
|
799
704
|
continue;
|
|
800
705
|
}
|
|
801
|
-
|
|
802
|
-
|
|
803
|
-
|
|
706
|
+
const className = pointer.className;
|
|
707
|
+
// only include the good pointers
|
|
804
708
|
if (className) {
|
|
805
709
|
pointersHash[className] = pointersHash[className] || new Set();
|
|
806
710
|
pointersHash[className].add(pointer.objectId);
|
|
807
711
|
}
|
|
808
712
|
}
|
|
809
|
-
|
|
810
713
|
const includeRestOptions = {};
|
|
811
|
-
|
|
812
714
|
if (restOptions.keys) {
|
|
813
715
|
const keys = new Set(restOptions.keys.split(','));
|
|
814
716
|
const keySet = Array.from(keys).reduce((set, key) => {
|
|
815
717
|
const keyPath = key.split('.');
|
|
816
718
|
let i = 0;
|
|
817
|
-
|
|
818
719
|
for (i; i < path.length; i++) {
|
|
819
720
|
if (path[i] != keyPath[i]) {
|
|
820
721
|
return set;
|
|
821
722
|
}
|
|
822
723
|
}
|
|
823
|
-
|
|
824
724
|
if (i < keyPath.length) {
|
|
825
725
|
set.add(keyPath[i]);
|
|
826
726
|
}
|
|
827
|
-
|
|
828
727
|
return set;
|
|
829
728
|
}, new Set());
|
|
830
|
-
|
|
831
729
|
if (keySet.size > 0) {
|
|
832
730
|
includeRestOptions.keys = Array.from(keySet).join(',');
|
|
833
731
|
}
|
|
834
732
|
}
|
|
835
|
-
|
|
836
733
|
if (restOptions.excludeKeys) {
|
|
837
734
|
const excludeKeys = new Set(restOptions.excludeKeys.split(','));
|
|
838
735
|
const excludeKeySet = Array.from(excludeKeys).reduce((set, key) => {
|
|
839
736
|
const keyPath = key.split('.');
|
|
840
737
|
let i = 0;
|
|
841
|
-
|
|
842
738
|
for (i; i < path.length; i++) {
|
|
843
739
|
if (path[i] != keyPath[i]) {
|
|
844
740
|
return set;
|
|
845
741
|
}
|
|
846
742
|
}
|
|
847
|
-
|
|
848
743
|
if (i == keyPath.length - 1) {
|
|
849
744
|
set.add(keyPath[i]);
|
|
850
745
|
}
|
|
851
|
-
|
|
852
746
|
return set;
|
|
853
747
|
}, new Set());
|
|
854
|
-
|
|
855
748
|
if (excludeKeySet.size > 0) {
|
|
856
749
|
includeRestOptions.excludeKeys = Array.from(excludeKeySet).join(',');
|
|
857
750
|
}
|
|
858
751
|
}
|
|
859
|
-
|
|
860
752
|
if (restOptions.includeReadPreference) {
|
|
861
753
|
includeRestOptions.readPreference = restOptions.includeReadPreference;
|
|
862
754
|
includeRestOptions.includeReadPreference = restOptions.includeReadPreference;
|
|
863
755
|
} else if (restOptions.readPreference) {
|
|
864
756
|
includeRestOptions.readPreference = restOptions.readPreference;
|
|
865
757
|
}
|
|
866
|
-
|
|
867
758
|
const queryPromises = Object.keys(pointersHash).map(className => {
|
|
868
759
|
const objectIds = Array.from(pointersHash[className]);
|
|
869
760
|
let where;
|
|
870
|
-
|
|
871
761
|
if (objectIds.length === 1) {
|
|
872
762
|
where = {
|
|
873
763
|
objectId: objectIds[0]
|
|
@@ -879,7 +769,6 @@ function includePath(config, auth, response, path, restOptions = {}) {
|
|
|
879
769
|
}
|
|
880
770
|
};
|
|
881
771
|
}
|
|
882
|
-
|
|
883
772
|
var query = new RestQuery(config, auth, className, where, includeRestOptions);
|
|
884
773
|
return query.execute({
|
|
885
774
|
op: 'get'
|
|
@@ -887,105 +776,86 @@ function includePath(config, auth, response, path, restOptions = {}) {
|
|
|
887
776
|
results.className = className;
|
|
888
777
|
return Promise.resolve(results);
|
|
889
778
|
});
|
|
890
|
-
});
|
|
779
|
+
});
|
|
891
780
|
|
|
781
|
+
// Get the objects for all these object ids
|
|
892
782
|
return Promise.all(queryPromises).then(responses => {
|
|
893
783
|
var replace = responses.reduce((replace, includeResponse) => {
|
|
894
784
|
for (var obj of includeResponse.results) {
|
|
895
785
|
obj.__type = 'Object';
|
|
896
786
|
obj.className = includeResponse.className;
|
|
897
|
-
|
|
898
787
|
if (obj.className == '_User' && !auth.isMaster) {
|
|
899
788
|
delete obj.sessionToken;
|
|
900
789
|
delete obj.authData;
|
|
901
790
|
}
|
|
902
|
-
|
|
903
791
|
replace[obj.objectId] = obj;
|
|
904
792
|
}
|
|
905
|
-
|
|
906
793
|
return replace;
|
|
907
794
|
}, {});
|
|
908
795
|
var resp = {
|
|
909
796
|
results: replacePointers(response.results, path, replace)
|
|
910
797
|
};
|
|
911
|
-
|
|
912
798
|
if (response.count) {
|
|
913
799
|
resp.count = response.count;
|
|
914
800
|
}
|
|
915
|
-
|
|
916
801
|
return resp;
|
|
917
802
|
});
|
|
918
|
-
}
|
|
803
|
+
}
|
|
804
|
+
|
|
805
|
+
// Object may be a list of REST-format object to find pointers in, or
|
|
919
806
|
// it may be a single object.
|
|
920
807
|
// If the path yields things that aren't pointers, this throws an error.
|
|
921
808
|
// Path is a list of fields to search into.
|
|
922
809
|
// Returns a list of pointers in REST format.
|
|
923
|
-
|
|
924
|
-
|
|
925
810
|
function findPointers(object, path) {
|
|
926
811
|
if (object instanceof Array) {
|
|
927
812
|
var answer = [];
|
|
928
|
-
|
|
929
813
|
for (var x of object) {
|
|
930
814
|
answer = answer.concat(findPointers(x, path));
|
|
931
815
|
}
|
|
932
|
-
|
|
933
816
|
return answer;
|
|
934
817
|
}
|
|
935
|
-
|
|
936
818
|
if (typeof object !== 'object' || !object) {
|
|
937
819
|
return [];
|
|
938
820
|
}
|
|
939
|
-
|
|
940
821
|
if (path.length == 0) {
|
|
941
822
|
if (object === null || object.__type == 'Pointer') {
|
|
942
823
|
return [object];
|
|
943
824
|
}
|
|
944
|
-
|
|
945
825
|
return [];
|
|
946
826
|
}
|
|
947
|
-
|
|
948
827
|
var subobject = object[path[0]];
|
|
949
|
-
|
|
950
828
|
if (!subobject) {
|
|
951
829
|
return [];
|
|
952
830
|
}
|
|
953
|
-
|
|
954
831
|
return findPointers(subobject, path.slice(1));
|
|
955
|
-
}
|
|
832
|
+
}
|
|
833
|
+
|
|
834
|
+
// Object may be a list of REST-format objects to replace pointers
|
|
956
835
|
// in, or it may be a single object.
|
|
957
836
|
// Path is a list of fields to search into.
|
|
958
837
|
// replace is a map from object id -> object.
|
|
959
838
|
// Returns something analogous to object, but with the appropriate
|
|
960
839
|
// pointers inflated.
|
|
961
|
-
|
|
962
|
-
|
|
963
840
|
function replacePointers(object, path, replace) {
|
|
964
841
|
if (object instanceof Array) {
|
|
965
842
|
return object.map(obj => replacePointers(obj, path, replace)).filter(obj => typeof obj !== 'undefined');
|
|
966
843
|
}
|
|
967
|
-
|
|
968
844
|
if (typeof object !== 'object' || !object) {
|
|
969
845
|
return object;
|
|
970
846
|
}
|
|
971
|
-
|
|
972
847
|
if (path.length === 0) {
|
|
973
848
|
if (object && object.__type === 'Pointer') {
|
|
974
849
|
return replace[object.objectId];
|
|
975
850
|
}
|
|
976
|
-
|
|
977
851
|
return object;
|
|
978
852
|
}
|
|
979
|
-
|
|
980
853
|
var subobject = object[path[0]];
|
|
981
|
-
|
|
982
854
|
if (!subobject) {
|
|
983
855
|
return object;
|
|
984
856
|
}
|
|
985
|
-
|
|
986
857
|
var newsub = replacePointers(subobject, path.slice(1), replace);
|
|
987
858
|
var answer = {};
|
|
988
|
-
|
|
989
859
|
for (var key in object) {
|
|
990
860
|
if (key == path[0]) {
|
|
991
861
|
answer[key] = newsub;
|
|
@@ -993,39 +863,32 @@ function replacePointers(object, path, replace) {
|
|
|
993
863
|
answer[key] = object[key];
|
|
994
864
|
}
|
|
995
865
|
}
|
|
996
|
-
|
|
997
866
|
return answer;
|
|
998
|
-
}
|
|
999
|
-
// Returns undefined otherwise.
|
|
1000
|
-
|
|
867
|
+
}
|
|
1001
868
|
|
|
869
|
+
// Finds a subobject that has the given key, if there is one.
|
|
870
|
+
// Returns undefined otherwise.
|
|
1002
871
|
function findObjectWithKey(root, key) {
|
|
1003
872
|
if (typeof root !== 'object') {
|
|
1004
873
|
return;
|
|
1005
874
|
}
|
|
1006
|
-
|
|
1007
875
|
if (root instanceof Array) {
|
|
1008
876
|
for (var item of root) {
|
|
1009
877
|
const answer = findObjectWithKey(item, key);
|
|
1010
|
-
|
|
1011
878
|
if (answer) {
|
|
1012
879
|
return answer;
|
|
1013
880
|
}
|
|
1014
881
|
}
|
|
1015
882
|
}
|
|
1016
|
-
|
|
1017
883
|
if (root && root[key]) {
|
|
1018
884
|
return root;
|
|
1019
885
|
}
|
|
1020
|
-
|
|
1021
886
|
for (var subkey in root) {
|
|
1022
887
|
const answer = findObjectWithKey(root[subkey], key);
|
|
1023
|
-
|
|
1024
888
|
if (answer) {
|
|
1025
889
|
return answer;
|
|
1026
890
|
}
|
|
1027
891
|
}
|
|
1028
892
|
}
|
|
1029
|
-
|
|
1030
893
|
module.exports = RestQuery;
|
|
1031
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/RestQuery.js"],"names":["SchemaController","require","Parse","triggers","continueWhile","AlwaysSelectedKeys","RestQuery","config","auth","className","restWhere","restOptions","clientSDK","runAfterFind","context","response","findOptions","isMaster","user","Error","INVALID_SESSION_TOKEN","$and","__type","objectId","id","doCount","includeAll","include","keysForInclude","Object","prototype","hasOwnProperty","call","keys","excludeKeys","length","split","filter","key","map","slice","lastIndexOf","join","option","concat","Array","from","Set","exclude","k","indexOf","fields","order","sort","reduce","sortMap","field","trim","score","$meta","paths","includes","pathSet","memo","path","index","parts","s","a","b","redirectKey","redirectClassNameForKey","redirectClassName","INVALID_JSON","execute","executeOptions","Promise","resolve","then","buildRestWhere","denyProtectedFields","handleIncludeAll","handleExcludeKeys","runFind","runCount","handleInclude","runAfterFindTrigger","each","callback","limit","finished","query","results","forEach","assign","$gt","getUserAndRoleACL","validateClientClassCreation","replaceSelect","replaceDontSelect","replaceInQuery","replaceNotInQuery","replaceEquality","acl","getUserRoles","roles","database","newClassName","allowClientClassCreation","systemClasses","loadSchema","schemaController","hasClass","OPERATION_FORBIDDEN","transformInQuery","inQueryObject","values","result","push","isArray","findObjectWithKey","inQueryValue","where","INVALID_QUERY","additionalOptions","subqueryReadPreference","readPreference","subquery","transformNotInQuery","notInQueryObject","notInQueryValue","getDeepestObjectFromKey","json","idx","src","splice","transformSelect","selectObject","objects","selectValue","transformDontSelect","dontSelectObject","dontSelectValue","cleanResultAuthData","password","authData","provider","replaceEqualityConstraint","constraint","equalToObject","hasDirectConstraint","hasOperatorConstraint","options","op","find","explain","filesController","expandFilesInObject","r","count","skip","c","protectedFields","addProtectedFields","getOneSchema","schema","includeFields","keyFields","type","pathResponse","includePath","newResponse","hasAfterFindHook","triggerExists","Types","afterFind","applicationId","pipeline","distinct","parseQuery","Query","withJSON","maybeRunAfterFindTrigger","object","toJSON","pointers","findPointers","pointersHash","pointer","add","includeRestOptions","keySet","set","keyPath","i","size","excludeKeySet","includeReadPreference","queryPromises","objectIds","$in","all","responses","replace","includeResponse","obj","sessionToken","resp","replacePointers","answer","x","subobject","newsub","root","item","subkey","module","exports"],"mappings":"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","sourcesContent":["// An object that encapsulates everything we need to run a 'find'\n// operation, encoded in the REST API format.\n\nvar SchemaController = require('./Controllers/SchemaController');\nvar Parse = require('parse/node').Parse;\nconst triggers = require('./triggers');\nconst { continueWhile } = require('parse/lib/node/promiseUtils');\nconst AlwaysSelectedKeys = ['objectId', 'createdAt', 'updatedAt', 'ACL'];\n// restOptions can include:\n//   skip\n//   limit\n//   order\n//   count\n//   include\n//   keys\n//   excludeKeys\n//   redirectClassNameForKey\n//   readPreference\n//   includeReadPreference\n//   subqueryReadPreference\nfunction RestQuery(\n  config,\n  auth,\n  className,\n  restWhere = {},\n  restOptions = {},\n  clientSDK,\n  runAfterFind = true,\n  context\n) {\n  this.config = config;\n  this.auth = auth;\n  this.className = className;\n  this.restWhere = restWhere;\n  this.restOptions = restOptions;\n  this.clientSDK = clientSDK;\n  this.runAfterFind = runAfterFind;\n  this.response = null;\n  this.findOptions = {};\n  this.context = context || {};\n  if (!this.auth.isMaster) {\n    if (this.className == '_Session') {\n      if (!this.auth.user) {\n        throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n      }\n      this.restWhere = {\n        $and: [\n          this.restWhere,\n          {\n            user: {\n              __type: 'Pointer',\n              className: '_User',\n              objectId: this.auth.user.id,\n            },\n          },\n        ],\n      };\n    }\n  }\n\n  this.doCount = false;\n  this.includeAll = false;\n\n  // The format for this.include is not the same as the format for the\n  // include option - it's the paths we should include, in order,\n  // stored as arrays, taking into account that we need to include foo\n  // before including foo.bar. Also it should dedupe.\n  // For example, passing an arg of include=foo.bar,foo.baz could lead to\n  // this.include = [['foo'], ['foo', 'baz'], ['foo', 'bar']]\n  this.include = [];\n  let keysForInclude = '';\n\n  // If we have keys, we probably want to force some includes (n-1 level)\n  // See issue: https://github.com/parse-community/parse-server/issues/3185\n  if (Object.prototype.hasOwnProperty.call(restOptions, 'keys')) {\n    keysForInclude = restOptions.keys;\n  }\n\n  // If we have keys, we probably want to force some includes (n-1 level)\n  // in order to exclude specific keys.\n  if (Object.prototype.hasOwnProperty.call(restOptions, 'excludeKeys')) {\n    keysForInclude += ',' + restOptions.excludeKeys;\n  }\n\n  if (keysForInclude.length > 0) {\n    keysForInclude = keysForInclude\n      .split(',')\n      .filter(key => {\n        // At least 2 components\n        return key.split('.').length > 1;\n      })\n      .map(key => {\n        // Slice the last component (a.b.c -> a.b)\n        // Otherwise we'll include one level too much.\n        return key.slice(0, key.lastIndexOf('.'));\n      })\n      .join(',');\n\n    // Concat the possibly present include string with the one from the keys\n    // Dedup / sorting is handle in 'include' case.\n    if (keysForInclude.length > 0) {\n      if (!restOptions.include || restOptions.include.length == 0) {\n        restOptions.include = keysForInclude;\n      } else {\n        restOptions.include += ',' + keysForInclude;\n      }\n    }\n  }\n\n  for (var option in restOptions) {\n    switch (option) {\n      case 'keys': {\n        const keys = restOptions.keys\n          .split(',')\n          .filter(key => key.length > 0)\n          .concat(AlwaysSelectedKeys);\n        this.keys = Array.from(new Set(keys));\n        break;\n      }\n      case 'excludeKeys': {\n        const exclude = restOptions.excludeKeys\n          .split(',')\n          .filter(k => AlwaysSelectedKeys.indexOf(k) < 0);\n        this.excludeKeys = Array.from(new Set(exclude));\n        break;\n      }\n      case 'count':\n        this.doCount = true;\n        break;\n      case 'includeAll':\n        this.includeAll = true;\n        break;\n      case 'explain':\n      case 'hint':\n      case 'distinct':\n      case 'pipeline':\n      case 'skip':\n      case 'limit':\n      case 'readPreference':\n        this.findOptions[option] = restOptions[option];\n        break;\n      case 'order':\n        var fields = restOptions.order.split(',');\n        this.findOptions.sort = fields.reduce((sortMap, field) => {\n          field = field.trim();\n          if (field === '$score' || field === '-$score') {\n            sortMap.score = { $meta: 'textScore' };\n          } else if (field[0] == '-') {\n            sortMap[field.slice(1)] = -1;\n          } else {\n            sortMap[field] = 1;\n          }\n          return sortMap;\n        }, {});\n        break;\n      case 'include': {\n        const paths = restOptions.include.split(',');\n        if (paths.includes('*')) {\n          this.includeAll = true;\n          break;\n        }\n        // Load the existing includes (from keys)\n        const pathSet = paths.reduce((memo, path) => {\n          // Split each paths on . (a.b.c -> [a,b,c])\n          // reduce to create all paths\n          // ([a,b,c] -> {a: true, 'a.b': true, 'a.b.c': true})\n          return path.split('.').reduce((memo, path, index, parts) => {\n            memo[parts.slice(0, index + 1).join('.')] = true;\n            return memo;\n          }, memo);\n        }, {});\n\n        this.include = Object.keys(pathSet)\n          .map(s => {\n            return s.split('.');\n          })\n          .sort((a, b) => {\n            return a.length - b.length; // Sort by number of components\n          });\n        break;\n      }\n      case 'redirectClassNameForKey':\n        this.redirectKey = restOptions.redirectClassNameForKey;\n        this.redirectClassName = null;\n        break;\n      case 'includeReadPreference':\n      case 'subqueryReadPreference':\n        break;\n      default:\n        throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad option: ' + option);\n    }\n  }\n}\n\n// A convenient method to perform all the steps of processing a query\n// in order.\n// Returns a promise for the response - an object with optional keys\n// 'results' and 'count'.\n// TODO: consolidate the replaceX functions\nRestQuery.prototype.execute = function (executeOptions) {\n  return Promise.resolve()\n    .then(() => {\n      return this.buildRestWhere();\n    })\n    .then(() => {\n      return this.denyProtectedFields();\n    })\n    .then(() => {\n      return this.handleIncludeAll();\n    })\n    .then(() => {\n      return this.handleExcludeKeys();\n    })\n    .then(() => {\n      return this.runFind(executeOptions);\n    })\n    .then(() => {\n      return this.runCount();\n    })\n    .then(() => {\n      return this.handleInclude();\n    })\n    .then(() => {\n      return this.runAfterFindTrigger();\n    })\n    .then(() => {\n      return this.response;\n    });\n};\n\nRestQuery.prototype.each = function (callback) {\n  const { config, auth, className, restWhere, restOptions, clientSDK } = this;\n  // if the limit is set, use it\n  restOptions.limit = restOptions.limit || 100;\n  restOptions.order = 'objectId';\n  let finished = false;\n\n  return continueWhile(\n    () => {\n      return !finished;\n    },\n    async () => {\n      const query = new RestQuery(\n        config,\n        auth,\n        className,\n        restWhere,\n        restOptions,\n        clientSDK,\n        this.runAfterFind,\n        this.context\n      );\n      const { results } = await query.execute();\n      results.forEach(callback);\n      finished = results.length < restOptions.limit;\n      if (!finished) {\n        restWhere.objectId = Object.assign({}, restWhere.objectId, {\n          $gt: results[results.length - 1].objectId,\n        });\n      }\n    }\n  );\n};\n\nRestQuery.prototype.buildRestWhere = function () {\n  return Promise.resolve()\n    .then(() => {\n      return this.getUserAndRoleACL();\n    })\n    .then(() => {\n      return this.redirectClassNameForKey();\n    })\n    .then(() => {\n      return this.validateClientClassCreation();\n    })\n    .then(() => {\n      return this.replaceSelect();\n    })\n    .then(() => {\n      return this.replaceDontSelect();\n    })\n    .then(() => {\n      return this.replaceInQuery();\n    })\n    .then(() => {\n      return this.replaceNotInQuery();\n    })\n    .then(() => {\n      return this.replaceEquality();\n    });\n};\n\n// Uses the Auth object to get the list of roles, adds the user id\nRestQuery.prototype.getUserAndRoleACL = function () {\n  if (this.auth.isMaster) {\n    return Promise.resolve();\n  }\n\n  this.findOptions.acl = ['*'];\n\n  if (this.auth.user) {\n    return this.auth.getUserRoles().then(roles => {\n      this.findOptions.acl = this.findOptions.acl.concat(roles, [this.auth.user.id]);\n      return;\n    });\n  } else {\n    return Promise.resolve();\n  }\n};\n\n// Changes the className if redirectClassNameForKey is set.\n// Returns a promise.\nRestQuery.prototype.redirectClassNameForKey = function () {\n  if (!this.redirectKey) {\n    return Promise.resolve();\n  }\n\n  // We need to change the class name based on the schema\n  return this.config.database\n    .redirectClassNameForKey(this.className, this.redirectKey)\n    .then(newClassName => {\n      this.className = newClassName;\n      this.redirectClassName = newClassName;\n    });\n};\n\n// Validates this operation against the allowClientClassCreation config.\nRestQuery.prototype.validateClientClassCreation = function () {\n  if (\n    this.config.allowClientClassCreation === false &&\n    !this.auth.isMaster &&\n    SchemaController.systemClasses.indexOf(this.className) === -1\n  ) {\n    return this.config.database\n      .loadSchema()\n      .then(schemaController => schemaController.hasClass(this.className))\n      .then(hasClass => {\n        if (hasClass !== true) {\n          throw new Parse.Error(\n            Parse.Error.OPERATION_FORBIDDEN,\n            'This user is not allowed to access ' + 'non-existent class: ' + this.className\n          );\n        }\n      });\n  } else {\n    return Promise.resolve();\n  }\n};\n\nfunction transformInQuery(inQueryObject, className, results) {\n  var values = [];\n  for (var result of results) {\n    values.push({\n      __type: 'Pointer',\n      className: className,\n      objectId: result.objectId,\n    });\n  }\n  delete inQueryObject['$inQuery'];\n  if (Array.isArray(inQueryObject['$in'])) {\n    inQueryObject['$in'] = inQueryObject['$in'].concat(values);\n  } else {\n    inQueryObject['$in'] = values;\n  }\n}\n\n// Replaces a $inQuery clause by running the subquery, if there is an\n// $inQuery clause.\n// The $inQuery clause turns into an $in with values that are just\n// pointers to the objects returned in the subquery.\nRestQuery.prototype.replaceInQuery = function () {\n  var inQueryObject = findObjectWithKey(this.restWhere, '$inQuery');\n  if (!inQueryObject) {\n    return;\n  }\n\n  // The inQuery value must have precisely two keys - where and className\n  var inQueryValue = inQueryObject['$inQuery'];\n  if (!inQueryValue.where || !inQueryValue.className) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $inQuery');\n  }\n\n  const additionalOptions = {\n    redirectClassNameForKey: inQueryValue.redirectClassNameForKey,\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  } else if (this.restOptions.readPreference) {\n    additionalOptions.readPreference = this.restOptions.readPreference;\n  }\n\n  var subquery = new RestQuery(\n    this.config,\n    this.auth,\n    inQueryValue.className,\n    inQueryValue.where,\n    additionalOptions\n  );\n  return subquery.execute().then(response => {\n    transformInQuery(inQueryObject, subquery.className, response.results);\n    // Recurse to repeat\n    return this.replaceInQuery();\n  });\n};\n\nfunction transformNotInQuery(notInQueryObject, className, results) {\n  var values = [];\n  for (var result of results) {\n    values.push({\n      __type: 'Pointer',\n      className: className,\n      objectId: result.objectId,\n    });\n  }\n  delete notInQueryObject['$notInQuery'];\n  if (Array.isArray(notInQueryObject['$nin'])) {\n    notInQueryObject['$nin'] = notInQueryObject['$nin'].concat(values);\n  } else {\n    notInQueryObject['$nin'] = values;\n  }\n}\n\n// Replaces a $notInQuery clause by running the subquery, if there is an\n// $notInQuery clause.\n// The $notInQuery clause turns into a $nin with values that are just\n// pointers to the objects returned in the subquery.\nRestQuery.prototype.replaceNotInQuery = function () {\n  var notInQueryObject = findObjectWithKey(this.restWhere, '$notInQuery');\n  if (!notInQueryObject) {\n    return;\n  }\n\n  // The notInQuery value must have precisely two keys - where and className\n  var notInQueryValue = notInQueryObject['$notInQuery'];\n  if (!notInQueryValue.where || !notInQueryValue.className) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $notInQuery');\n  }\n\n  const additionalOptions = {\n    redirectClassNameForKey: notInQueryValue.redirectClassNameForKey,\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  } else if (this.restOptions.readPreference) {\n    additionalOptions.readPreference = this.restOptions.readPreference;\n  }\n\n  var subquery = new RestQuery(\n    this.config,\n    this.auth,\n    notInQueryValue.className,\n    notInQueryValue.where,\n    additionalOptions\n  );\n  return subquery.execute().then(response => {\n    transformNotInQuery(notInQueryObject, subquery.className, response.results);\n    // Recurse to repeat\n    return this.replaceNotInQuery();\n  });\n};\n\n// Used to get the deepest object from json using dot notation.\nconst getDeepestObjectFromKey = (json, key, idx, src) => {\n  if (key in json) {\n    return json[key];\n  }\n  src.splice(1); // Exit Early\n};\n\nconst transformSelect = (selectObject, key, objects) => {\n  var values = [];\n  for (var result of objects) {\n    values.push(key.split('.').reduce(getDeepestObjectFromKey, result));\n  }\n  delete selectObject['$select'];\n  if (Array.isArray(selectObject['$in'])) {\n    selectObject['$in'] = selectObject['$in'].concat(values);\n  } else {\n    selectObject['$in'] = values;\n  }\n};\n\n// Replaces a $select clause by running the subquery, if there is a\n// $select clause.\n// The $select clause turns into an $in with values selected out of\n// the subquery.\n// Returns a possible-promise.\nRestQuery.prototype.replaceSelect = function () {\n  var selectObject = findObjectWithKey(this.restWhere, '$select');\n  if (!selectObject) {\n    return;\n  }\n\n  // The select value must have precisely two keys - query and key\n  var selectValue = selectObject['$select'];\n  // iOS SDK don't send where if not set, let it pass\n  if (\n    !selectValue.query ||\n    !selectValue.key ||\n    typeof selectValue.query !== 'object' ||\n    !selectValue.query.className ||\n    Object.keys(selectValue).length !== 2\n  ) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $select');\n  }\n\n  const additionalOptions = {\n    redirectClassNameForKey: selectValue.query.redirectClassNameForKey,\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  } else if (this.restOptions.readPreference) {\n    additionalOptions.readPreference = this.restOptions.readPreference;\n  }\n\n  var subquery = new RestQuery(\n    this.config,\n    this.auth,\n    selectValue.query.className,\n    selectValue.query.where,\n    additionalOptions\n  );\n  return subquery.execute().then(response => {\n    transformSelect(selectObject, selectValue.key, response.results);\n    // Keep replacing $select clauses\n    return this.replaceSelect();\n  });\n};\n\nconst transformDontSelect = (dontSelectObject, key, objects) => {\n  var values = [];\n  for (var result of objects) {\n    values.push(key.split('.').reduce(getDeepestObjectFromKey, result));\n  }\n  delete dontSelectObject['$dontSelect'];\n  if (Array.isArray(dontSelectObject['$nin'])) {\n    dontSelectObject['$nin'] = dontSelectObject['$nin'].concat(values);\n  } else {\n    dontSelectObject['$nin'] = values;\n  }\n};\n\n// Replaces a $dontSelect clause by running the subquery, if there is a\n// $dontSelect clause.\n// The $dontSelect clause turns into an $nin with values selected out of\n// the subquery.\n// Returns a possible-promise.\nRestQuery.prototype.replaceDontSelect = function () {\n  var dontSelectObject = findObjectWithKey(this.restWhere, '$dontSelect');\n  if (!dontSelectObject) {\n    return;\n  }\n\n  // The dontSelect value must have precisely two keys - query and key\n  var dontSelectValue = dontSelectObject['$dontSelect'];\n  if (\n    !dontSelectValue.query ||\n    !dontSelectValue.key ||\n    typeof dontSelectValue.query !== 'object' ||\n    !dontSelectValue.query.className ||\n    Object.keys(dontSelectValue).length !== 2\n  ) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $dontSelect');\n  }\n  const additionalOptions = {\n    redirectClassNameForKey: dontSelectValue.query.redirectClassNameForKey,\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  } else if (this.restOptions.readPreference) {\n    additionalOptions.readPreference = this.restOptions.readPreference;\n  }\n\n  var subquery = new RestQuery(\n    this.config,\n    this.auth,\n    dontSelectValue.query.className,\n    dontSelectValue.query.where,\n    additionalOptions\n  );\n  return subquery.execute().then(response => {\n    transformDontSelect(dontSelectObject, dontSelectValue.key, response.results);\n    // Keep replacing $dontSelect clauses\n    return this.replaceDontSelect();\n  });\n};\n\nRestQuery.prototype.cleanResultAuthData = function (result) {\n  delete result.password;\n  if (result.authData) {\n    Object.keys(result.authData).forEach(provider => {\n      if (result.authData[provider] === null) {\n        delete result.authData[provider];\n      }\n    });\n\n    if (Object.keys(result.authData).length == 0) {\n      delete result.authData;\n    }\n  }\n};\n\nconst replaceEqualityConstraint = constraint => {\n  if (typeof constraint !== 'object') {\n    return constraint;\n  }\n  const equalToObject = {};\n  let hasDirectConstraint = false;\n  let hasOperatorConstraint = false;\n  for (const key in constraint) {\n    if (key.indexOf('$') !== 0) {\n      hasDirectConstraint = true;\n      equalToObject[key] = constraint[key];\n    } else {\n      hasOperatorConstraint = true;\n    }\n  }\n  if (hasDirectConstraint && hasOperatorConstraint) {\n    constraint['$eq'] = equalToObject;\n    Object.keys(equalToObject).forEach(key => {\n      delete constraint[key];\n    });\n  }\n  return constraint;\n};\n\nRestQuery.prototype.replaceEquality = function () {\n  if (typeof this.restWhere !== 'object') {\n    return;\n  }\n  for (const key in this.restWhere) {\n    this.restWhere[key] = replaceEqualityConstraint(this.restWhere[key]);\n  }\n};\n\n// Returns a promise for whether it was successful.\n// Populates this.response with an object that only has 'results'.\nRestQuery.prototype.runFind = function (options = {}) {\n  if (this.findOptions.limit === 0) {\n    this.response = { results: [] };\n    return Promise.resolve();\n  }\n  const findOptions = Object.assign({}, this.findOptions);\n  if (this.keys) {\n    findOptions.keys = this.keys.map(key => {\n      return key.split('.')[0];\n    });\n  }\n  if (options.op) {\n    findOptions.op = options.op;\n  }\n  return this.config.database\n    .find(this.className, this.restWhere, findOptions, this.auth)\n    .then(results => {\n      if (this.className === '_User' && !findOptions.explain) {\n        for (var result of results) {\n          this.cleanResultAuthData(result);\n        }\n      }\n\n      this.config.filesController.expandFilesInObject(this.config, results);\n\n      if (this.redirectClassName) {\n        for (var r of results) {\n          r.className = this.redirectClassName;\n        }\n      }\n      this.response = { results: results };\n    });\n};\n\n// Returns a promise for whether it was successful.\n// Populates this.response.count with the count\nRestQuery.prototype.runCount = function () {\n  if (!this.doCount) {\n    return;\n  }\n  this.findOptions.count = true;\n  delete this.findOptions.skip;\n  delete this.findOptions.limit;\n  return this.config.database.find(this.className, this.restWhere, this.findOptions).then(c => {\n    this.response.count = c;\n  });\n};\n\nRestQuery.prototype.denyProtectedFields = async function () {\n  if (this.auth.isMaster) {\n    return;\n  }\n  const schemaController = await this.config.database.loadSchema();\n  const protectedFields =\n    this.config.database.addProtectedFields(\n      schemaController,\n      this.className,\n      this.restWhere,\n      this.findOptions.acl,\n      this.auth,\n      this.findOptions\n    ) || [];\n  for (const key of protectedFields) {\n    if (this.restWhere[key]) {\n      throw new Parse.Error(\n        Parse.Error.OPERATION_FORBIDDEN,\n        `This user is not allowed to query ${key} on class ${this.className}`\n      );\n    }\n  }\n};\n\n// Augments this.response with all pointers on an object\nRestQuery.prototype.handleIncludeAll = function () {\n  if (!this.includeAll) {\n    return;\n  }\n  return this.config.database\n    .loadSchema()\n    .then(schemaController => schemaController.getOneSchema(this.className))\n    .then(schema => {\n      const includeFields = [];\n      const keyFields = [];\n      for (const field in schema.fields) {\n        if (\n          (schema.fields[field].type && schema.fields[field].type === 'Pointer') ||\n          (schema.fields[field].type && schema.fields[field].type === 'Array')\n        ) {\n          includeFields.push([field]);\n          keyFields.push(field);\n        }\n      }\n      // Add fields to include, keys, remove dups\n      this.include = [...new Set([...this.include, ...includeFields])];\n      // if this.keys not set, then all keys are already included\n      if (this.keys) {\n        this.keys = [...new Set([...this.keys, ...keyFields])];\n      }\n    });\n};\n\n// Updates property `this.keys` to contain all keys but the ones unselected.\nRestQuery.prototype.handleExcludeKeys = function () {\n  if (!this.excludeKeys) {\n    return;\n  }\n  if (this.keys) {\n    this.keys = this.keys.filter(k => !this.excludeKeys.includes(k));\n    return;\n  }\n  return this.config.database\n    .loadSchema()\n    .then(schemaController => schemaController.getOneSchema(this.className))\n    .then(schema => {\n      const fields = Object.keys(schema.fields);\n      this.keys = fields.filter(k => !this.excludeKeys.includes(k));\n    });\n};\n\n// Augments this.response with data at the paths provided in this.include.\nRestQuery.prototype.handleInclude = function () {\n  if (this.include.length == 0) {\n    return;\n  }\n\n  var pathResponse = includePath(\n    this.config,\n    this.auth,\n    this.response,\n    this.include[0],\n    this.restOptions\n  );\n  if (pathResponse.then) {\n    return pathResponse.then(newResponse => {\n      this.response = newResponse;\n      this.include = this.include.slice(1);\n      return this.handleInclude();\n    });\n  } else if (this.include.length > 0) {\n    this.include = this.include.slice(1);\n    return this.handleInclude();\n  }\n\n  return pathResponse;\n};\n\n//Returns a promise of a processed set of results\nRestQuery.prototype.runAfterFindTrigger = function () {\n  if (!this.response) {\n    return;\n  }\n  if (!this.runAfterFind) {\n    return;\n  }\n  // Avoid doing any setup for triggers if there is no 'afterFind' trigger for this class.\n  const hasAfterFindHook = triggers.triggerExists(\n    this.className,\n    triggers.Types.afterFind,\n    this.config.applicationId\n  );\n  if (!hasAfterFindHook) {\n    return Promise.resolve();\n  }\n  // Skip Aggregate and Distinct Queries\n  if (this.findOptions.pipeline || this.findOptions.distinct) {\n    return Promise.resolve();\n  }\n\n  const json = Object.assign({}, this.restOptions);\n  json.where = this.restWhere;\n  const parseQuery = new Parse.Query(this.className);\n  parseQuery.withJSON(json);\n  // Run afterFind trigger and set the new results\n  return triggers\n    .maybeRunAfterFindTrigger(\n      triggers.Types.afterFind,\n      this.auth,\n      this.className,\n      this.response.results,\n      this.config,\n      parseQuery,\n      this.context\n    )\n    .then(results => {\n      // Ensure we properly set the className back\n      if (this.redirectClassName) {\n        this.response.results = results.map(object => {\n          if (object instanceof Parse.Object) {\n            object = object.toJSON();\n          }\n          object.className = this.redirectClassName;\n          return object;\n        });\n      } else {\n        this.response.results = results;\n      }\n    });\n};\n\n// Adds included values to the response.\n// Path is a list of field names.\n// Returns a promise for an augmented response.\nfunction includePath(config, auth, response, path, restOptions = {}) {\n  var pointers = findPointers(response.results, path);\n  if (pointers.length == 0) {\n    return response;\n  }\n  const pointersHash = {};\n  for (var pointer of pointers) {\n    if (!pointer) {\n      continue;\n    }\n    const className = pointer.className;\n    // only include the good pointers\n    if (className) {\n      pointersHash[className] = pointersHash[className] || new Set();\n      pointersHash[className].add(pointer.objectId);\n    }\n  }\n  const includeRestOptions = {};\n  if (restOptions.keys) {\n    const keys = new Set(restOptions.keys.split(','));\n    const keySet = Array.from(keys).reduce((set, key) => {\n      const keyPath = key.split('.');\n      let i = 0;\n      for (i; i < path.length; i++) {\n        if (path[i] != keyPath[i]) {\n          return set;\n        }\n      }\n      if (i < keyPath.length) {\n        set.add(keyPath[i]);\n      }\n      return set;\n    }, new Set());\n    if (keySet.size > 0) {\n      includeRestOptions.keys = Array.from(keySet).join(',');\n    }\n  }\n\n  if (restOptions.excludeKeys) {\n    const excludeKeys = new Set(restOptions.excludeKeys.split(','));\n    const excludeKeySet = Array.from(excludeKeys).reduce((set, key) => {\n      const keyPath = key.split('.');\n      let i = 0;\n      for (i; i < path.length; i++) {\n        if (path[i] != keyPath[i]) {\n          return set;\n        }\n      }\n      if (i == keyPath.length - 1) {\n        set.add(keyPath[i]);\n      }\n      return set;\n    }, new Set());\n    if (excludeKeySet.size > 0) {\n      includeRestOptions.excludeKeys = Array.from(excludeKeySet).join(',');\n    }\n  }\n\n  if (restOptions.includeReadPreference) {\n    includeRestOptions.readPreference = restOptions.includeReadPreference;\n    includeRestOptions.includeReadPreference = restOptions.includeReadPreference;\n  } else if (restOptions.readPreference) {\n    includeRestOptions.readPreference = restOptions.readPreference;\n  }\n\n  const queryPromises = Object.keys(pointersHash).map(className => {\n    const objectIds = Array.from(pointersHash[className]);\n    let where;\n    if (objectIds.length === 1) {\n      where = { objectId: objectIds[0] };\n    } else {\n      where = { objectId: { $in: objectIds } };\n    }\n    var query = new RestQuery(config, auth, className, where, includeRestOptions);\n    return query.execute({ op: 'get' }).then(results => {\n      results.className = className;\n      return Promise.resolve(results);\n    });\n  });\n\n  // Get the objects for all these object ids\n  return Promise.all(queryPromises).then(responses => {\n    var replace = responses.reduce((replace, includeResponse) => {\n      for (var obj of includeResponse.results) {\n        obj.__type = 'Object';\n        obj.className = includeResponse.className;\n\n        if (obj.className == '_User' && !auth.isMaster) {\n          delete obj.sessionToken;\n          delete obj.authData;\n        }\n        replace[obj.objectId] = obj;\n      }\n      return replace;\n    }, {});\n\n    var resp = {\n      results: replacePointers(response.results, path, replace),\n    };\n    if (response.count) {\n      resp.count = response.count;\n    }\n    return resp;\n  });\n}\n\n// Object may be a list of REST-format object to find pointers in, or\n// it may be a single object.\n// If the path yields things that aren't pointers, this throws an error.\n// Path is a list of fields to search into.\n// Returns a list of pointers in REST format.\nfunction findPointers(object, path) {\n  if (object instanceof Array) {\n    var answer = [];\n    for (var x of object) {\n      answer = answer.concat(findPointers(x, path));\n    }\n    return answer;\n  }\n\n  if (typeof object !== 'object' || !object) {\n    return [];\n  }\n\n  if (path.length == 0) {\n    if (object === null || object.__type == 'Pointer') {\n      return [object];\n    }\n    return [];\n  }\n\n  var subobject = object[path[0]];\n  if (!subobject) {\n    return [];\n  }\n  return findPointers(subobject, path.slice(1));\n}\n\n// Object may be a list of REST-format objects to replace pointers\n// in, or it may be a single object.\n// Path is a list of fields to search into.\n// replace is a map from object id -> object.\n// Returns something analogous to object, but with the appropriate\n// pointers inflated.\nfunction replacePointers(object, path, replace) {\n  if (object instanceof Array) {\n    return object\n      .map(obj => replacePointers(obj, path, replace))\n      .filter(obj => typeof obj !== 'undefined');\n  }\n\n  if (typeof object !== 'object' || !object) {\n    return object;\n  }\n\n  if (path.length === 0) {\n    if (object && object.__type === 'Pointer') {\n      return replace[object.objectId];\n    }\n    return object;\n  }\n\n  var subobject = object[path[0]];\n  if (!subobject) {\n    return object;\n  }\n  var newsub = replacePointers(subobject, path.slice(1), replace);\n  var answer = {};\n  for (var key in object) {\n    if (key == path[0]) {\n      answer[key] = newsub;\n    } else {\n      answer[key] = object[key];\n    }\n  }\n  return answer;\n}\n\n// Finds a subobject that has the given key, if there is one.\n// Returns undefined otherwise.\nfunction findObjectWithKey(root, key) {\n  if (typeof root !== 'object') {\n    return;\n  }\n  if (root instanceof Array) {\n    for (var item of root) {\n      const answer = findObjectWithKey(item, key);\n      if (answer) {\n        return answer;\n      }\n    }\n  }\n  if (root && root[key]) {\n    return root;\n  }\n  for (var subkey in root) {\n    const answer = findObjectWithKey(root[subkey], key);\n    if (answer) {\n      return answer;\n    }\n  }\n}\n\nmodule.exports = RestQuery;\n"]}
|
|
894
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["SchemaController","require","Parse","triggers","continueWhile","AlwaysSelectedKeys","RestQuery","config","auth","className","restWhere","restOptions","clientSDK","runAfterFind","context","response","findOptions","isMaster","user","Error","INVALID_SESSION_TOKEN","$and","__type","objectId","id","doCount","includeAll","include","keysForInclude","Object","prototype","hasOwnProperty","call","keys","excludeKeys","length","split","filter","key","map","slice","lastIndexOf","join","option","concat","Array","from","Set","exclude","k","indexOf","fields","order","sort","reduce","sortMap","field","trim","score","$meta","paths","includes","pathSet","memo","path","index","parts","s","a","b","redirectKey","redirectClassNameForKey","redirectClassName","INVALID_JSON","execute","executeOptions","Promise","resolve","then","buildRestWhere","denyProtectedFields","handleIncludeAll","handleExcludeKeys","runFind","runCount","handleInclude","runAfterFindTrigger","each","callback","limit","finished","query","results","forEach","assign","$gt","getUserAndRoleACL","validateClientClassCreation","replaceSelect","replaceDontSelect","replaceInQuery","replaceNotInQuery","replaceEquality","acl","getUserRoles","roles","database","newClassName","allowClientClassCreation","systemClasses","loadSchema","schemaController","hasClass","OPERATION_FORBIDDEN","transformInQuery","inQueryObject","values","result","push","isArray","findObjectWithKey","inQueryValue","where","INVALID_QUERY","additionalOptions","subqueryReadPreference","readPreference","subquery","transformNotInQuery","notInQueryObject","notInQueryValue","getDeepestObjectFromKey","json","idx","src","splice","transformSelect","selectObject","objects","selectValue","transformDontSelect","dontSelectObject","dontSelectValue","cleanResultAuthData","password","authData","provider","replaceEqualityConstraint","constraint","equalToObject","hasDirectConstraint","hasOperatorConstraint","options","op","find","explain","filesController","expandFilesInObject","r","count","skip","c","protectedFields","addProtectedFields","getOneSchema","schema","includeFields","keyFields","type","pathResponse","includePath","newResponse","hasAfterFindHook","triggerExists","Types","afterFind","applicationId","pipeline","distinct","parseQuery","Query","withJSON","maybeRunAfterFindTrigger","object","toJSON","pointers","findPointers","pointersHash","pointer","add","includeRestOptions","keySet","set","keyPath","i","size","excludeKeySet","includeReadPreference","queryPromises","objectIds","$in","all","responses","replace","includeResponse","obj","sessionToken","resp","replacePointers","answer","x","subobject","newsub","root","item","subkey","module","exports"],"sources":["../src/RestQuery.js"],"sourcesContent":["// An object that encapsulates everything we need to run a 'find'\n// operation, encoded in the REST API format.\n\nvar SchemaController = require('./Controllers/SchemaController');\nvar Parse = require('parse/node').Parse;\nconst triggers = require('./triggers');\nconst { continueWhile } = require('parse/lib/node/promiseUtils');\nconst AlwaysSelectedKeys = ['objectId', 'createdAt', 'updatedAt', 'ACL'];\n// restOptions can include:\n//   skip\n//   limit\n//   order\n//   count\n//   include\n//   keys\n//   excludeKeys\n//   redirectClassNameForKey\n//   readPreference\n//   includeReadPreference\n//   subqueryReadPreference\nfunction RestQuery(\n  config,\n  auth,\n  className,\n  restWhere = {},\n  restOptions = {},\n  clientSDK,\n  runAfterFind = true,\n  context\n) {\n  this.config = config;\n  this.auth = auth;\n  this.className = className;\n  this.restWhere = restWhere;\n  this.restOptions = restOptions;\n  this.clientSDK = clientSDK;\n  this.runAfterFind = runAfterFind;\n  this.response = null;\n  this.findOptions = {};\n  this.context = context || {};\n  if (!this.auth.isMaster) {\n    if (this.className == '_Session') {\n      if (!this.auth.user) {\n        throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n      }\n      this.restWhere = {\n        $and: [\n          this.restWhere,\n          {\n            user: {\n              __type: 'Pointer',\n              className: '_User',\n              objectId: this.auth.user.id,\n            },\n          },\n        ],\n      };\n    }\n  }\n\n  this.doCount = false;\n  this.includeAll = false;\n\n  // The format for this.include is not the same as the format for the\n  // include option - it's the paths we should include, in order,\n  // stored as arrays, taking into account that we need to include foo\n  // before including foo.bar. Also it should dedupe.\n  // For example, passing an arg of include=foo.bar,foo.baz could lead to\n  // this.include = [['foo'], ['foo', 'baz'], ['foo', 'bar']]\n  this.include = [];\n  let keysForInclude = '';\n\n  // If we have keys, we probably want to force some includes (n-1 level)\n  // See issue: https://github.com/parse-community/parse-server/issues/3185\n  if (Object.prototype.hasOwnProperty.call(restOptions, 'keys')) {\n    keysForInclude = restOptions.keys;\n  }\n\n  // If we have keys, we probably want to force some includes (n-1 level)\n  // in order to exclude specific keys.\n  if (Object.prototype.hasOwnProperty.call(restOptions, 'excludeKeys')) {\n    keysForInclude += ',' + restOptions.excludeKeys;\n  }\n\n  if (keysForInclude.length > 0) {\n    keysForInclude = keysForInclude\n      .split(',')\n      .filter(key => {\n        // At least 2 components\n        return key.split('.').length > 1;\n      })\n      .map(key => {\n        // Slice the last component (a.b.c -> a.b)\n        // Otherwise we'll include one level too much.\n        return key.slice(0, key.lastIndexOf('.'));\n      })\n      .join(',');\n\n    // Concat the possibly present include string with the one from the keys\n    // Dedup / sorting is handle in 'include' case.\n    if (keysForInclude.length > 0) {\n      if (!restOptions.include || restOptions.include.length == 0) {\n        restOptions.include = keysForInclude;\n      } else {\n        restOptions.include += ',' + keysForInclude;\n      }\n    }\n  }\n\n  for (var option in restOptions) {\n    switch (option) {\n      case 'keys': {\n        const keys = restOptions.keys\n          .split(',')\n          .filter(key => key.length > 0)\n          .concat(AlwaysSelectedKeys);\n        this.keys = Array.from(new Set(keys));\n        break;\n      }\n      case 'excludeKeys': {\n        const exclude = restOptions.excludeKeys\n          .split(',')\n          .filter(k => AlwaysSelectedKeys.indexOf(k) < 0);\n        this.excludeKeys = Array.from(new Set(exclude));\n        break;\n      }\n      case 'count':\n        this.doCount = true;\n        break;\n      case 'includeAll':\n        this.includeAll = true;\n        break;\n      case 'explain':\n      case 'hint':\n      case 'distinct':\n      case 'pipeline':\n      case 'skip':\n      case 'limit':\n      case 'readPreference':\n        this.findOptions[option] = restOptions[option];\n        break;\n      case 'order':\n        var fields = restOptions.order.split(',');\n        this.findOptions.sort = fields.reduce((sortMap, field) => {\n          field = field.trim();\n          if (field === '$score' || field === '-$score') {\n            sortMap.score = { $meta: 'textScore' };\n          } else if (field[0] == '-') {\n            sortMap[field.slice(1)] = -1;\n          } else {\n            sortMap[field] = 1;\n          }\n          return sortMap;\n        }, {});\n        break;\n      case 'include': {\n        const paths = restOptions.include.split(',');\n        if (paths.includes('*')) {\n          this.includeAll = true;\n          break;\n        }\n        // Load the existing includes (from keys)\n        const pathSet = paths.reduce((memo, path) => {\n          // Split each paths on . (a.b.c -> [a,b,c])\n          // reduce to create all paths\n          // ([a,b,c] -> {a: true, 'a.b': true, 'a.b.c': true})\n          return path.split('.').reduce((memo, path, index, parts) => {\n            memo[parts.slice(0, index + 1).join('.')] = true;\n            return memo;\n          }, memo);\n        }, {});\n\n        this.include = Object.keys(pathSet)\n          .map(s => {\n            return s.split('.');\n          })\n          .sort((a, b) => {\n            return a.length - b.length; // Sort by number of components\n          });\n        break;\n      }\n      case 'redirectClassNameForKey':\n        this.redirectKey = restOptions.redirectClassNameForKey;\n        this.redirectClassName = null;\n        break;\n      case 'includeReadPreference':\n      case 'subqueryReadPreference':\n        break;\n      default:\n        throw new Parse.Error(Parse.Error.INVALID_JSON, 'bad option: ' + option);\n    }\n  }\n}\n\n// A convenient method to perform all the steps of processing a query\n// in order.\n// Returns a promise for the response - an object with optional keys\n// 'results' and 'count'.\n// TODO: consolidate the replaceX functions\nRestQuery.prototype.execute = function (executeOptions) {\n  return Promise.resolve()\n    .then(() => {\n      return this.buildRestWhere();\n    })\n    .then(() => {\n      return this.denyProtectedFields();\n    })\n    .then(() => {\n      return this.handleIncludeAll();\n    })\n    .then(() => {\n      return this.handleExcludeKeys();\n    })\n    .then(() => {\n      return this.runFind(executeOptions);\n    })\n    .then(() => {\n      return this.runCount();\n    })\n    .then(() => {\n      return this.handleInclude();\n    })\n    .then(() => {\n      return this.runAfterFindTrigger();\n    })\n    .then(() => {\n      return this.response;\n    });\n};\n\nRestQuery.prototype.each = function (callback) {\n  const { config, auth, className, restWhere, restOptions, clientSDK } = this;\n  // if the limit is set, use it\n  restOptions.limit = restOptions.limit || 100;\n  restOptions.order = 'objectId';\n  let finished = false;\n\n  return continueWhile(\n    () => {\n      return !finished;\n    },\n    async () => {\n      const query = new RestQuery(\n        config,\n        auth,\n        className,\n        restWhere,\n        restOptions,\n        clientSDK,\n        this.runAfterFind,\n        this.context\n      );\n      const { results } = await query.execute();\n      results.forEach(callback);\n      finished = results.length < restOptions.limit;\n      if (!finished) {\n        restWhere.objectId = Object.assign({}, restWhere.objectId, {\n          $gt: results[results.length - 1].objectId,\n        });\n      }\n    }\n  );\n};\n\nRestQuery.prototype.buildRestWhere = function () {\n  return Promise.resolve()\n    .then(() => {\n      return this.getUserAndRoleACL();\n    })\n    .then(() => {\n      return this.redirectClassNameForKey();\n    })\n    .then(() => {\n      return this.validateClientClassCreation();\n    })\n    .then(() => {\n      return this.replaceSelect();\n    })\n    .then(() => {\n      return this.replaceDontSelect();\n    })\n    .then(() => {\n      return this.replaceInQuery();\n    })\n    .then(() => {\n      return this.replaceNotInQuery();\n    })\n    .then(() => {\n      return this.replaceEquality();\n    });\n};\n\n// Uses the Auth object to get the list of roles, adds the user id\nRestQuery.prototype.getUserAndRoleACL = function () {\n  if (this.auth.isMaster) {\n    return Promise.resolve();\n  }\n\n  this.findOptions.acl = ['*'];\n\n  if (this.auth.user) {\n    return this.auth.getUserRoles().then(roles => {\n      this.findOptions.acl = this.findOptions.acl.concat(roles, [this.auth.user.id]);\n      return;\n    });\n  } else {\n    return Promise.resolve();\n  }\n};\n\n// Changes the className if redirectClassNameForKey is set.\n// Returns a promise.\nRestQuery.prototype.redirectClassNameForKey = function () {\n  if (!this.redirectKey) {\n    return Promise.resolve();\n  }\n\n  // We need to change the class name based on the schema\n  return this.config.database\n    .redirectClassNameForKey(this.className, this.redirectKey)\n    .then(newClassName => {\n      this.className = newClassName;\n      this.redirectClassName = newClassName;\n    });\n};\n\n// Validates this operation against the allowClientClassCreation config.\nRestQuery.prototype.validateClientClassCreation = function () {\n  if (\n    this.config.allowClientClassCreation === false &&\n    !this.auth.isMaster &&\n    SchemaController.systemClasses.indexOf(this.className) === -1\n  ) {\n    return this.config.database\n      .loadSchema()\n      .then(schemaController => schemaController.hasClass(this.className))\n      .then(hasClass => {\n        if (hasClass !== true) {\n          throw new Parse.Error(\n            Parse.Error.OPERATION_FORBIDDEN,\n            'This user is not allowed to access ' + 'non-existent class: ' + this.className\n          );\n        }\n      });\n  } else {\n    return Promise.resolve();\n  }\n};\n\nfunction transformInQuery(inQueryObject, className, results) {\n  var values = [];\n  for (var result of results) {\n    values.push({\n      __type: 'Pointer',\n      className: className,\n      objectId: result.objectId,\n    });\n  }\n  delete inQueryObject['$inQuery'];\n  if (Array.isArray(inQueryObject['$in'])) {\n    inQueryObject['$in'] = inQueryObject['$in'].concat(values);\n  } else {\n    inQueryObject['$in'] = values;\n  }\n}\n\n// Replaces a $inQuery clause by running the subquery, if there is an\n// $inQuery clause.\n// The $inQuery clause turns into an $in with values that are just\n// pointers to the objects returned in the subquery.\nRestQuery.prototype.replaceInQuery = function () {\n  var inQueryObject = findObjectWithKey(this.restWhere, '$inQuery');\n  if (!inQueryObject) {\n    return;\n  }\n\n  // The inQuery value must have precisely two keys - where and className\n  var inQueryValue = inQueryObject['$inQuery'];\n  if (!inQueryValue.where || !inQueryValue.className) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $inQuery');\n  }\n\n  const additionalOptions = {\n    redirectClassNameForKey: inQueryValue.redirectClassNameForKey,\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  } else if (this.restOptions.readPreference) {\n    additionalOptions.readPreference = this.restOptions.readPreference;\n  }\n\n  var subquery = new RestQuery(\n    this.config,\n    this.auth,\n    inQueryValue.className,\n    inQueryValue.where,\n    additionalOptions\n  );\n  return subquery.execute().then(response => {\n    transformInQuery(inQueryObject, subquery.className, response.results);\n    // Recurse to repeat\n    return this.replaceInQuery();\n  });\n};\n\nfunction transformNotInQuery(notInQueryObject, className, results) {\n  var values = [];\n  for (var result of results) {\n    values.push({\n      __type: 'Pointer',\n      className: className,\n      objectId: result.objectId,\n    });\n  }\n  delete notInQueryObject['$notInQuery'];\n  if (Array.isArray(notInQueryObject['$nin'])) {\n    notInQueryObject['$nin'] = notInQueryObject['$nin'].concat(values);\n  } else {\n    notInQueryObject['$nin'] = values;\n  }\n}\n\n// Replaces a $notInQuery clause by running the subquery, if there is an\n// $notInQuery clause.\n// The $notInQuery clause turns into a $nin with values that are just\n// pointers to the objects returned in the subquery.\nRestQuery.prototype.replaceNotInQuery = function () {\n  var notInQueryObject = findObjectWithKey(this.restWhere, '$notInQuery');\n  if (!notInQueryObject) {\n    return;\n  }\n\n  // The notInQuery value must have precisely two keys - where and className\n  var notInQueryValue = notInQueryObject['$notInQuery'];\n  if (!notInQueryValue.where || !notInQueryValue.className) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $notInQuery');\n  }\n\n  const additionalOptions = {\n    redirectClassNameForKey: notInQueryValue.redirectClassNameForKey,\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  } else if (this.restOptions.readPreference) {\n    additionalOptions.readPreference = this.restOptions.readPreference;\n  }\n\n  var subquery = new RestQuery(\n    this.config,\n    this.auth,\n    notInQueryValue.className,\n    notInQueryValue.where,\n    additionalOptions\n  );\n  return subquery.execute().then(response => {\n    transformNotInQuery(notInQueryObject, subquery.className, response.results);\n    // Recurse to repeat\n    return this.replaceNotInQuery();\n  });\n};\n\n// Used to get the deepest object from json using dot notation.\nconst getDeepestObjectFromKey = (json, key, idx, src) => {\n  if (key in json) {\n    return json[key];\n  }\n  src.splice(1); // Exit Early\n};\n\nconst transformSelect = (selectObject, key, objects) => {\n  var values = [];\n  for (var result of objects) {\n    values.push(key.split('.').reduce(getDeepestObjectFromKey, result));\n  }\n  delete selectObject['$select'];\n  if (Array.isArray(selectObject['$in'])) {\n    selectObject['$in'] = selectObject['$in'].concat(values);\n  } else {\n    selectObject['$in'] = values;\n  }\n};\n\n// Replaces a $select clause by running the subquery, if there is a\n// $select clause.\n// The $select clause turns into an $in with values selected out of\n// the subquery.\n// Returns a possible-promise.\nRestQuery.prototype.replaceSelect = function () {\n  var selectObject = findObjectWithKey(this.restWhere, '$select');\n  if (!selectObject) {\n    return;\n  }\n\n  // The select value must have precisely two keys - query and key\n  var selectValue = selectObject['$select'];\n  // iOS SDK don't send where if not set, let it pass\n  if (\n    !selectValue.query ||\n    !selectValue.key ||\n    typeof selectValue.query !== 'object' ||\n    !selectValue.query.className ||\n    Object.keys(selectValue).length !== 2\n  ) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $select');\n  }\n\n  const additionalOptions = {\n    redirectClassNameForKey: selectValue.query.redirectClassNameForKey,\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  } else if (this.restOptions.readPreference) {\n    additionalOptions.readPreference = this.restOptions.readPreference;\n  }\n\n  var subquery = new RestQuery(\n    this.config,\n    this.auth,\n    selectValue.query.className,\n    selectValue.query.where,\n    additionalOptions\n  );\n  return subquery.execute().then(response => {\n    transformSelect(selectObject, selectValue.key, response.results);\n    // Keep replacing $select clauses\n    return this.replaceSelect();\n  });\n};\n\nconst transformDontSelect = (dontSelectObject, key, objects) => {\n  var values = [];\n  for (var result of objects) {\n    values.push(key.split('.').reduce(getDeepestObjectFromKey, result));\n  }\n  delete dontSelectObject['$dontSelect'];\n  if (Array.isArray(dontSelectObject['$nin'])) {\n    dontSelectObject['$nin'] = dontSelectObject['$nin'].concat(values);\n  } else {\n    dontSelectObject['$nin'] = values;\n  }\n};\n\n// Replaces a $dontSelect clause by running the subquery, if there is a\n// $dontSelect clause.\n// The $dontSelect clause turns into an $nin with values selected out of\n// the subquery.\n// Returns a possible-promise.\nRestQuery.prototype.replaceDontSelect = function () {\n  var dontSelectObject = findObjectWithKey(this.restWhere, '$dontSelect');\n  if (!dontSelectObject) {\n    return;\n  }\n\n  // The dontSelect value must have precisely two keys - query and key\n  var dontSelectValue = dontSelectObject['$dontSelect'];\n  if (\n    !dontSelectValue.query ||\n    !dontSelectValue.key ||\n    typeof dontSelectValue.query !== 'object' ||\n    !dontSelectValue.query.className ||\n    Object.keys(dontSelectValue).length !== 2\n  ) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'improper usage of $dontSelect');\n  }\n  const additionalOptions = {\n    redirectClassNameForKey: dontSelectValue.query.redirectClassNameForKey,\n  };\n\n  if (this.restOptions.subqueryReadPreference) {\n    additionalOptions.readPreference = this.restOptions.subqueryReadPreference;\n    additionalOptions.subqueryReadPreference = this.restOptions.subqueryReadPreference;\n  } else if (this.restOptions.readPreference) {\n    additionalOptions.readPreference = this.restOptions.readPreference;\n  }\n\n  var subquery = new RestQuery(\n    this.config,\n    this.auth,\n    dontSelectValue.query.className,\n    dontSelectValue.query.where,\n    additionalOptions\n  );\n  return subquery.execute().then(response => {\n    transformDontSelect(dontSelectObject, dontSelectValue.key, response.results);\n    // Keep replacing $dontSelect clauses\n    return this.replaceDontSelect();\n  });\n};\n\nRestQuery.prototype.cleanResultAuthData = function (result) {\n  delete result.password;\n  if (result.authData) {\n    Object.keys(result.authData).forEach(provider => {\n      if (result.authData[provider] === null) {\n        delete result.authData[provider];\n      }\n    });\n\n    if (Object.keys(result.authData).length == 0) {\n      delete result.authData;\n    }\n  }\n};\n\nconst replaceEqualityConstraint = constraint => {\n  if (typeof constraint !== 'object') {\n    return constraint;\n  }\n  const equalToObject = {};\n  let hasDirectConstraint = false;\n  let hasOperatorConstraint = false;\n  for (const key in constraint) {\n    if (key.indexOf('$') !== 0) {\n      hasDirectConstraint = true;\n      equalToObject[key] = constraint[key];\n    } else {\n      hasOperatorConstraint = true;\n    }\n  }\n  if (hasDirectConstraint && hasOperatorConstraint) {\n    constraint['$eq'] = equalToObject;\n    Object.keys(equalToObject).forEach(key => {\n      delete constraint[key];\n    });\n  }\n  return constraint;\n};\n\nRestQuery.prototype.replaceEquality = function () {\n  if (typeof this.restWhere !== 'object') {\n    return;\n  }\n  for (const key in this.restWhere) {\n    this.restWhere[key] = replaceEqualityConstraint(this.restWhere[key]);\n  }\n};\n\n// Returns a promise for whether it was successful.\n// Populates this.response with an object that only has 'results'.\nRestQuery.prototype.runFind = function (options = {}) {\n  if (this.findOptions.limit === 0) {\n    this.response = { results: [] };\n    return Promise.resolve();\n  }\n  const findOptions = Object.assign({}, this.findOptions);\n  if (this.keys) {\n    findOptions.keys = this.keys.map(key => {\n      return key.split('.')[0];\n    });\n  }\n  if (options.op) {\n    findOptions.op = options.op;\n  }\n  return this.config.database\n    .find(this.className, this.restWhere, findOptions, this.auth)\n    .then(results => {\n      if (this.className === '_User' && !findOptions.explain) {\n        for (var result of results) {\n          this.cleanResultAuthData(result);\n        }\n      }\n\n      this.config.filesController.expandFilesInObject(this.config, results);\n\n      if (this.redirectClassName) {\n        for (var r of results) {\n          r.className = this.redirectClassName;\n        }\n      }\n      this.response = { results: results };\n    });\n};\n\n// Returns a promise for whether it was successful.\n// Populates this.response.count with the count\nRestQuery.prototype.runCount = function () {\n  if (!this.doCount) {\n    return;\n  }\n  this.findOptions.count = true;\n  delete this.findOptions.skip;\n  delete this.findOptions.limit;\n  return this.config.database.find(this.className, this.restWhere, this.findOptions).then(c => {\n    this.response.count = c;\n  });\n};\n\nRestQuery.prototype.denyProtectedFields = async function () {\n  if (this.auth.isMaster) {\n    return;\n  }\n  const schemaController = await this.config.database.loadSchema();\n  const protectedFields =\n    this.config.database.addProtectedFields(\n      schemaController,\n      this.className,\n      this.restWhere,\n      this.findOptions.acl,\n      this.auth,\n      this.findOptions\n    ) || [];\n  for (const key of protectedFields) {\n    if (this.restWhere[key]) {\n      throw new Parse.Error(\n        Parse.Error.OPERATION_FORBIDDEN,\n        `This user is not allowed to query ${key} on class ${this.className}`\n      );\n    }\n  }\n};\n\n// Augments this.response with all pointers on an object\nRestQuery.prototype.handleIncludeAll = function () {\n  if (!this.includeAll) {\n    return;\n  }\n  return this.config.database\n    .loadSchema()\n    .then(schemaController => schemaController.getOneSchema(this.className))\n    .then(schema => {\n      const includeFields = [];\n      const keyFields = [];\n      for (const field in schema.fields) {\n        if (\n          (schema.fields[field].type && schema.fields[field].type === 'Pointer') ||\n          (schema.fields[field].type && schema.fields[field].type === 'Array')\n        ) {\n          includeFields.push([field]);\n          keyFields.push(field);\n        }\n      }\n      // Add fields to include, keys, remove dups\n      this.include = [...new Set([...this.include, ...includeFields])];\n      // if this.keys not set, then all keys are already included\n      if (this.keys) {\n        this.keys = [...new Set([...this.keys, ...keyFields])];\n      }\n    });\n};\n\n// Updates property `this.keys` to contain all keys but the ones unselected.\nRestQuery.prototype.handleExcludeKeys = function () {\n  if (!this.excludeKeys) {\n    return;\n  }\n  if (this.keys) {\n    this.keys = this.keys.filter(k => !this.excludeKeys.includes(k));\n    return;\n  }\n  return this.config.database\n    .loadSchema()\n    .then(schemaController => schemaController.getOneSchema(this.className))\n    .then(schema => {\n      const fields = Object.keys(schema.fields);\n      this.keys = fields.filter(k => !this.excludeKeys.includes(k));\n    });\n};\n\n// Augments this.response with data at the paths provided in this.include.\nRestQuery.prototype.handleInclude = function () {\n  if (this.include.length == 0) {\n    return;\n  }\n\n  var pathResponse = includePath(\n    this.config,\n    this.auth,\n    this.response,\n    this.include[0],\n    this.restOptions\n  );\n  if (pathResponse.then) {\n    return pathResponse.then(newResponse => {\n      this.response = newResponse;\n      this.include = this.include.slice(1);\n      return this.handleInclude();\n    });\n  } else if (this.include.length > 0) {\n    this.include = this.include.slice(1);\n    return this.handleInclude();\n  }\n\n  return pathResponse;\n};\n\n//Returns a promise of a processed set of results\nRestQuery.prototype.runAfterFindTrigger = function () {\n  if (!this.response) {\n    return;\n  }\n  if (!this.runAfterFind) {\n    return;\n  }\n  // Avoid doing any setup for triggers if there is no 'afterFind' trigger for this class.\n  const hasAfterFindHook = triggers.triggerExists(\n    this.className,\n    triggers.Types.afterFind,\n    this.config.applicationId\n  );\n  if (!hasAfterFindHook) {\n    return Promise.resolve();\n  }\n  // Skip Aggregate and Distinct Queries\n  if (this.findOptions.pipeline || this.findOptions.distinct) {\n    return Promise.resolve();\n  }\n\n  const json = Object.assign({}, this.restOptions);\n  json.where = this.restWhere;\n  const parseQuery = new Parse.Query(this.className);\n  parseQuery.withJSON(json);\n  // Run afterFind trigger and set the new results\n  return triggers\n    .maybeRunAfterFindTrigger(\n      triggers.Types.afterFind,\n      this.auth,\n      this.className,\n      this.response.results,\n      this.config,\n      parseQuery,\n      this.context\n    )\n    .then(results => {\n      // Ensure we properly set the className back\n      if (this.redirectClassName) {\n        this.response.results = results.map(object => {\n          if (object instanceof Parse.Object) {\n            object = object.toJSON();\n          }\n          object.className = this.redirectClassName;\n          return object;\n        });\n      } else {\n        this.response.results = results;\n      }\n    });\n};\n\n// Adds included values to the response.\n// Path is a list of field names.\n// Returns a promise for an augmented response.\nfunction includePath(config, auth, response, path, restOptions = {}) {\n  var pointers = findPointers(response.results, path);\n  if (pointers.length == 0) {\n    return response;\n  }\n  const pointersHash = {};\n  for (var pointer of pointers) {\n    if (!pointer) {\n      continue;\n    }\n    const className = pointer.className;\n    // only include the good pointers\n    if (className) {\n      pointersHash[className] = pointersHash[className] || new Set();\n      pointersHash[className].add(pointer.objectId);\n    }\n  }\n  const includeRestOptions = {};\n  if (restOptions.keys) {\n    const keys = new Set(restOptions.keys.split(','));\n    const keySet = Array.from(keys).reduce((set, key) => {\n      const keyPath = key.split('.');\n      let i = 0;\n      for (i; i < path.length; i++) {\n        if (path[i] != keyPath[i]) {\n          return set;\n        }\n      }\n      if (i < keyPath.length) {\n        set.add(keyPath[i]);\n      }\n      return set;\n    }, new Set());\n    if (keySet.size > 0) {\n      includeRestOptions.keys = Array.from(keySet).join(',');\n    }\n  }\n\n  if (restOptions.excludeKeys) {\n    const excludeKeys = new Set(restOptions.excludeKeys.split(','));\n    const excludeKeySet = Array.from(excludeKeys).reduce((set, key) => {\n      const keyPath = key.split('.');\n      let i = 0;\n      for (i; i < path.length; i++) {\n        if (path[i] != keyPath[i]) {\n          return set;\n        }\n      }\n      if (i == keyPath.length - 1) {\n        set.add(keyPath[i]);\n      }\n      return set;\n    }, new Set());\n    if (excludeKeySet.size > 0) {\n      includeRestOptions.excludeKeys = Array.from(excludeKeySet).join(',');\n    }\n  }\n\n  if (restOptions.includeReadPreference) {\n    includeRestOptions.readPreference = restOptions.includeReadPreference;\n    includeRestOptions.includeReadPreference = restOptions.includeReadPreference;\n  } else if (restOptions.readPreference) {\n    includeRestOptions.readPreference = restOptions.readPreference;\n  }\n\n  const queryPromises = Object.keys(pointersHash).map(className => {\n    const objectIds = Array.from(pointersHash[className]);\n    let where;\n    if (objectIds.length === 1) {\n      where = { objectId: objectIds[0] };\n    } else {\n      where = { objectId: { $in: objectIds } };\n    }\n    var query = new RestQuery(config, auth, className, where, includeRestOptions);\n    return query.execute({ op: 'get' }).then(results => {\n      results.className = className;\n      return Promise.resolve(results);\n    });\n  });\n\n  // Get the objects for all these object ids\n  return Promise.all(queryPromises).then(responses => {\n    var replace = responses.reduce((replace, includeResponse) => {\n      for (var obj of includeResponse.results) {\n        obj.__type = 'Object';\n        obj.className = includeResponse.className;\n\n        if (obj.className == '_User' && !auth.isMaster) {\n          delete obj.sessionToken;\n          delete obj.authData;\n        }\n        replace[obj.objectId] = obj;\n      }\n      return replace;\n    }, {});\n\n    var resp = {\n      results: replacePointers(response.results, path, replace),\n    };\n    if (response.count) {\n      resp.count = response.count;\n    }\n    return resp;\n  });\n}\n\n// Object may be a list of REST-format object to find pointers in, or\n// it may be a single object.\n// If the path yields things that aren't pointers, this throws an error.\n// Path is a list of fields to search into.\n// Returns a list of pointers in REST format.\nfunction findPointers(object, path) {\n  if (object instanceof Array) {\n    var answer = [];\n    for (var x of object) {\n      answer = answer.concat(findPointers(x, path));\n    }\n    return answer;\n  }\n\n  if (typeof object !== 'object' || !object) {\n    return [];\n  }\n\n  if (path.length == 0) {\n    if (object === null || object.__type == 'Pointer') {\n      return [object];\n    }\n    return [];\n  }\n\n  var subobject = object[path[0]];\n  if (!subobject) {\n    return [];\n  }\n  return findPointers(subobject, path.slice(1));\n}\n\n// Object may be a list of REST-format objects to replace pointers\n// in, or it may be a single object.\n// Path is a list of fields to search into.\n// replace is a map from object id -> object.\n// Returns something analogous to object, but with the appropriate\n// pointers inflated.\nfunction replacePointers(object, path, replace) {\n  if (object instanceof Array) {\n    return object\n      .map(obj => replacePointers(obj, path, replace))\n      .filter(obj => typeof obj !== 'undefined');\n  }\n\n  if (typeof object !== 'object' || !object) {\n    return object;\n  }\n\n  if (path.length === 0) {\n    if (object && object.__type === 'Pointer') {\n      return replace[object.objectId];\n    }\n    return object;\n  }\n\n  var subobject = object[path[0]];\n  if (!subobject) {\n    return object;\n  }\n  var newsub = replacePointers(subobject, path.slice(1), replace);\n  var answer = {};\n  for (var key in object) {\n    if (key == path[0]) {\n      answer[key] = newsub;\n    } else {\n      answer[key] = object[key];\n    }\n  }\n  return answer;\n}\n\n// Finds a subobject that has the given key, if there is one.\n// Returns undefined otherwise.\nfunction findObjectWithKey(root, key) {\n  if (typeof root !== 'object') {\n    return;\n  }\n  if (root instanceof Array) {\n    for (var item of root) {\n      const answer = findObjectWithKey(item, key);\n      if (answer) {\n        return answer;\n      }\n    }\n  }\n  if (root && root[key]) {\n    return root;\n  }\n  for (var subkey in root) {\n    const answer = findObjectWithKey(root[subkey], key);\n    if (answer) {\n      return answer;\n    }\n  }\n}\n\nmodule.exports = RestQuery;\n"],"mappings":"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"}
|