parse-server 4.10.13 → 4.10.14
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/lib/Controllers/DatabaseController.js +24 -29
- package/lib/RestQuery.js +18 -1
- package/package.json +1 -1
|
@@ -81,20 +81,17 @@ const transformObjectACL = (_ref) => {
|
|
|
81
81
|
return result;
|
|
82
82
|
};
|
|
83
83
|
|
|
84
|
-
const
|
|
84
|
+
const specialQueryKeys = ['$and', '$or', '$nor', '_rperm', '_wperm'];
|
|
85
|
+
const specialMasterQueryKeys = [...specialQueryKeys, '_email_verify_token', '_perishable_token', '_tombstone', '_email_verify_token_expires_at', '_failed_login_count', '_account_lockout_expires_at', '_password_changed_at', '_password_history'];
|
|
85
86
|
|
|
86
|
-
const
|
|
87
|
-
return specialQuerykeys.indexOf(key) >= 0;
|
|
88
|
-
};
|
|
89
|
-
|
|
90
|
-
const validateQuery = query => {
|
|
87
|
+
const validateQuery = (query, isMaster, update) => {
|
|
91
88
|
if (query.ACL) {
|
|
92
89
|
throw new _node.Parse.Error(_node.Parse.Error.INVALID_QUERY, 'Cannot query on ACL.');
|
|
93
90
|
}
|
|
94
91
|
|
|
95
92
|
if (query.$or) {
|
|
96
93
|
if (query.$or instanceof Array) {
|
|
97
|
-
query.$or.forEach(validateQuery);
|
|
94
|
+
query.$or.forEach(value => validateQuery(value, isMaster, update));
|
|
98
95
|
} else {
|
|
99
96
|
throw new _node.Parse.Error(_node.Parse.Error.INVALID_QUERY, 'Bad $or format - use an array value.');
|
|
100
97
|
}
|
|
@@ -102,7 +99,7 @@ const validateQuery = query => {
|
|
|
102
99
|
|
|
103
100
|
if (query.$and) {
|
|
104
101
|
if (query.$and instanceof Array) {
|
|
105
|
-
query.$and.forEach(validateQuery);
|
|
102
|
+
query.$and.forEach(value => validateQuery(value, isMaster, update));
|
|
106
103
|
} else {
|
|
107
104
|
throw new _node.Parse.Error(_node.Parse.Error.INVALID_QUERY, 'Bad $and format - use an array value.');
|
|
108
105
|
}
|
|
@@ -110,7 +107,7 @@ const validateQuery = query => {
|
|
|
110
107
|
|
|
111
108
|
if (query.$nor) {
|
|
112
109
|
if (query.$nor instanceof Array && query.$nor.length > 0) {
|
|
113
|
-
query.$nor.forEach(validateQuery);
|
|
110
|
+
query.$nor.forEach(value => validateQuery(value, isMaster, update));
|
|
114
111
|
} else {
|
|
115
112
|
throw new _node.Parse.Error(_node.Parse.Error.INVALID_QUERY, 'Bad $nor format - use an array of at least 1 value.');
|
|
116
113
|
}
|
|
@@ -125,7 +122,7 @@ const validateQuery = query => {
|
|
|
125
122
|
}
|
|
126
123
|
}
|
|
127
124
|
|
|
128
|
-
if (!
|
|
125
|
+
if (!key.match(/^[a-zA-Z][a-zA-Z0-9_\.]*$/) && (!specialQueryKeys.includes(key) && !isMaster && !update || update && isMaster && !specialMasterQueryKeys.includes(key))) {
|
|
129
126
|
throw new _node.Parse.Error(_node.Parse.Error.INVALID_KEY_NAME, `Invalid key name: ${key}`);
|
|
130
127
|
}
|
|
131
128
|
});
|
|
@@ -202,27 +199,25 @@ const filterSensitiveData = (isMaster, aclGroup, auth, operation, schema, classN
|
|
|
202
199
|
perms.protectedFields && perms.protectedFields.temporaryKeys && perms.protectedFields.temporaryKeys.forEach(k => delete object[k]);
|
|
203
200
|
}
|
|
204
201
|
|
|
205
|
-
if (
|
|
206
|
-
|
|
202
|
+
if (isUserClass) {
|
|
203
|
+
object.password = object._hashed_password;
|
|
204
|
+
delete object._hashed_password;
|
|
205
|
+
delete object.sessionToken;
|
|
207
206
|
}
|
|
208
207
|
|
|
209
|
-
object.password = object._hashed_password;
|
|
210
|
-
delete object._hashed_password;
|
|
211
|
-
delete object.sessionToken;
|
|
212
|
-
|
|
213
208
|
if (isMaster) {
|
|
214
209
|
return object;
|
|
215
210
|
}
|
|
216
211
|
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
212
|
+
for (const key in object) {
|
|
213
|
+
if (key.charAt(0) === '_') {
|
|
214
|
+
delete object[key];
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
if (!isUserClass) {
|
|
219
|
+
return object;
|
|
220
|
+
}
|
|
226
221
|
|
|
227
222
|
if (aclGroup.indexOf(object.objectId) > -1) {
|
|
228
223
|
return object;
|
|
@@ -494,7 +489,7 @@ class DatabaseController {
|
|
|
494
489
|
query = addWriteACL(query, acl);
|
|
495
490
|
}
|
|
496
491
|
|
|
497
|
-
validateQuery(query);
|
|
492
|
+
validateQuery(query, isMaster, true);
|
|
498
493
|
return schemaController.getOneSchema(className, true).catch(error => {
|
|
499
494
|
// If the schema doesn't exist, pretend it exists with no fields. This behavior
|
|
500
495
|
// will likely need revisiting.
|
|
@@ -697,7 +692,7 @@ class DatabaseController {
|
|
|
697
692
|
query = addWriteACL(query, acl);
|
|
698
693
|
}
|
|
699
694
|
|
|
700
|
-
validateQuery(query);
|
|
695
|
+
validateQuery(query, isMaster, false);
|
|
701
696
|
return schemaController.getOneSchema(className).catch(error => {
|
|
702
697
|
// If the schema doesn't exist, pretend it exists with no fields. This behavior
|
|
703
698
|
// will likely need revisiting.
|
|
@@ -1110,7 +1105,7 @@ class DatabaseController {
|
|
|
1110
1105
|
}
|
|
1111
1106
|
}
|
|
1112
1107
|
|
|
1113
|
-
validateQuery(query);
|
|
1108
|
+
validateQuery(query, isMaster, false);
|
|
1114
1109
|
|
|
1115
1110
|
if (count) {
|
|
1116
1111
|
if (!classExists) {
|
|
@@ -1500,4 +1495,4 @@ module.exports = DatabaseController; // Expose validateQuery for tests
|
|
|
1500
1495
|
|
|
1501
1496
|
module.exports._validateQuery = validateQuery;
|
|
1502
1497
|
module.exports.filterSensitiveData = filterSensitiveData;
|
|
1503
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Controllers/DatabaseController.js"],"names":["addWriteACL","query","acl","newQuery","_","cloneDeep","_wperm","$in","addReadACL","_rperm","transformObjectACL","ACL","result","entry","read","push","write","specialQuerykeys","isSpecialQueryKey","key","indexOf","validateQuery","Parse","Error","INVALID_QUERY","$or","Array","forEach","$and","$nor","length","Object","keys","$regex","$options","match","INVALID_KEY_NAME","filterSensitiveData","isMaster","aclGroup","auth","operation","schema","className","protectedFields","object","userId","user","id","perms","getClassLevelPermissions","isReadOperation","protectedFieldsPointerPerm","filter","startsWith","map","substring","value","newProtectedFields","overrideProtectedFields","pointerPerm","pointerPermIncludesUser","readUserFieldValue","isArray","some","objectId","fields","v","includes","isUserClass","k","temporaryKeys","password","_hashed_password","sessionToken","_email_verify_token","_perishable_token","_perishable_token_expires_at","_tombstone","_email_verify_token_expires_at","_failed_login_count","_account_lockout_expires_at","_password_changed_at","_password_history","authData","specialKeysForUpdate","isSpecialUpdateKey","joinTableName","flattenUpdateOperatorsForCreate","__op","amount","INVALID_JSON","objects","COMMAND_UNAVAILABLE","transformAuthData","provider","providerData","fieldName","type","untransformObjectACL","output","getRootFieldName","split","relationSchema","relatedId","owningId","DatabaseController","constructor","adapter","schemaCache","options","schemaPromise","_transactionalSession","collectionExists","classExists","purgeCollection","loadSchema","then","schemaController","getOneSchema","deleteObjectsByQuery","validateClassName","SchemaController","classNameIsValid","Promise","reject","INVALID_CLASS_NAME","resolve","clearCache","load","loadSchemaIfNeeded","redirectClassNameForKey","t","getExpectedType","targetClass","validateObject","runOptions","undefined","s","canAddField","update","many","upsert","addsField","skipSanitization","validateOnly","validSchemaController","originalQuery","originalUpdate","relationUpdates","validatePermission","collectRelationUpdates","addPointerPermissions","catch","error","rootFieldName","fieldNameIsValid","updateOperation","innerKey","INVALID_NESTED_KEY","find","OBJECT_NOT_FOUND","updateObjectsByQuery","upsertOneObject","findOneAndUpdate","handleRelationUpdates","_sanitizeDatabaseResult","ops","deleteMe","process","op","x","pending","addRelation","removeRelation","all","fromClassName","fromId","toId","doc","code","destroy","parseFormatSchema","create","originalObject","createdAt","iso","__type","updatedAt","enforceClassExists","createObject","convertSchemaToAdapterSchema","classSchema","schemaData","schemaFields","newKeys","field","action","deleteEverything","fast","deleteAllClasses","clear","relatedIds","queryOptions","skip","limit","sort","findOptions","canSortOnJoinTables","_id","results","owningIds","reduceInRelation","ors","aQuery","index","promises","queries","constraintKey","isNegation","r","q","ids","addNotInObjectIdsIds","addInObjectIdsIds","reduceRelationKeys","relatedTo","idsFromString","idsFromEq","idsFromIn","allIds","list","totalLength","reduce","memo","idsIntersection","intersect","big","$eq","idsFromNin","Set","$nin","count","distinct","pipeline","readPreference","hint","caseInsensitive","explain","_created_at","_updated_at","addProtectedFields","aggregate","INTERNAL_SERVER_ERROR","deleteSchema","deleteClass","wasParseCollection","relationFieldNames","name","testPermissionsForClassName","userACL","groupKey","permFields","pointerFields","userPointer","fieldDescriptor","fieldType","prototype","hasOwnProperty","call","queryClause","$all","assign","preserveKeys","serverOnlyKeys","authenticated","roles","userRoles","acc","protectedKeysSets","protectedKeys","next","createTransactionalSession","transactionalSession","commitTransactionalSession","abortTransactionalSession","performInitialization","requiredUserFields","defaultColumns","_Default","_User","requiredRoleFields","_Role","requiredIdempotencyFields","_Idempotency","userClassPromise","roleClassPromise","idempotencyClassPromise","MongoStorageAdapter","usernameUniqueness","ensureUniqueness","logger","warn","usernameCaseInsensitiveIndex","ensureIndex","emailUniqueness","emailCaseInsensitiveIndex","roleUniqueness","idempotencyRequestIdIndex","idempotencyExpireIndex","ttl","indexPromise","updateSchemaWithIndexes","adapterInit","VolatileClassesSchemas","_expandResultOnKeyPath","path","firstKey","nextPath","slice","join","requestKeywordDenylist","keyword","Utils","objectContainsKeyValue","JSON","stringify","response","keyUpdate","module","exports","_validateQuery"],"mappings":"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","sourcesContent":["﻿// @flow\n// A database adapter that works with data exported from the hosted\n// Parse database.\n\n// @flow-disable-next\nimport { Parse } from 'parse/node';\n// @flow-disable-next\nimport _ from 'lodash';\n// @flow-disable-next\nimport intersect from 'intersect';\n// @flow-disable-next\nimport deepcopy from 'deepcopy';\nimport logger from '../logger';\nimport Utils from '../Utils';\nimport * as SchemaController from './SchemaController';\nimport { StorageAdapter } from '../Adapters/Storage/StorageAdapter';\nimport type { ParseServerOptions } from '../Options';\nimport type { QueryOptions, FullQueryOptions } from '../Adapters/Storage/StorageAdapter';\n\nfunction addWriteACL(query, acl) {\n  const newQuery = _.cloneDeep(query);\n  //Can't be any existing '_wperm' query, we don't allow client queries on that, no need to $and\n  newQuery._wperm = { $in: [null, ...acl] };\n  return newQuery;\n}\n\nfunction addReadACL(query, acl) {\n  const newQuery = _.cloneDeep(query);\n  //Can't be any existing '_rperm' query, we don't allow client queries on that, no need to $and\n  newQuery._rperm = { $in: [null, '*', ...acl] };\n  return newQuery;\n}\n\n// Transforms a REST API formatted ACL object to our two-field mongo format.\nconst transformObjectACL = ({ ACL, ...result }) => {\n  if (!ACL) {\n    return result;\n  }\n\n  result._wperm = [];\n  result._rperm = [];\n\n  for (const entry in ACL) {\n    if (ACL[entry].read) {\n      result._rperm.push(entry);\n    }\n    if (ACL[entry].write) {\n      result._wperm.push(entry);\n    }\n  }\n  return result;\n};\n\nconst specialQuerykeys = [\n  '$and',\n  '$or',\n  '$nor',\n  '_rperm',\n  '_wperm',\n  '_perishable_token',\n  '_email_verify_token',\n  '_email_verify_token_expires_at',\n  '_account_lockout_expires_at',\n  '_failed_login_count',\n];\n\nconst isSpecialQueryKey = key => {\n  return specialQuerykeys.indexOf(key) >= 0;\n};\n\nconst validateQuery = (query: any): void => {\n  if (query.ACL) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Cannot query on ACL.');\n  }\n\n  if (query.$or) {\n    if (query.$or instanceof Array) {\n      query.$or.forEach(validateQuery);\n    } else {\n      throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Bad $or format - use an array value.');\n    }\n  }\n\n  if (query.$and) {\n    if (query.$and instanceof Array) {\n      query.$and.forEach(validateQuery);\n    } else {\n      throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Bad $and format - use an array value.');\n    }\n  }\n\n  if (query.$nor) {\n    if (query.$nor instanceof Array && query.$nor.length > 0) {\n      query.$nor.forEach(validateQuery);\n    } else {\n      throw new Parse.Error(\n        Parse.Error.INVALID_QUERY,\n        'Bad $nor format - use an array of at least 1 value.'\n      );\n    }\n  }\n\n  Object.keys(query).forEach(key => {\n    if (query && query[key] && query[key].$regex) {\n      if (typeof query[key].$options === 'string') {\n        if (!query[key].$options.match(/^[imxs]+$/)) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_QUERY,\n            `Bad $options value for query: ${query[key].$options}`\n          );\n        }\n      }\n    }\n    if (!isSpecialQueryKey(key) && !key.match(/^[a-zA-Z][a-zA-Z0-9_\\.]*$/)) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Invalid key name: ${key}`);\n    }\n  });\n};\n\n// Filters out any data that shouldn't be on this REST-formatted object.\nconst filterSensitiveData = (\n  isMaster: boolean,\n  aclGroup: any[],\n  auth: any,\n  operation: any,\n  schema: SchemaController.SchemaController | any,\n  className: string,\n  protectedFields: null | Array<any>,\n  object: any\n) => {\n  let userId = null;\n  if (auth && auth.user) userId = auth.user.id;\n\n  // replace protectedFields when using pointer-permissions\n  const perms =\n    schema && schema.getClassLevelPermissions ? schema.getClassLevelPermissions(className) : {};\n  if (perms) {\n    const isReadOperation = ['get', 'find'].indexOf(operation) > -1;\n\n    if (isReadOperation && perms.protectedFields) {\n      // extract protectedFields added with the pointer-permission prefix\n      const protectedFieldsPointerPerm = Object.keys(perms.protectedFields)\n        .filter(key => key.startsWith('userField:'))\n        .map(key => {\n          return { key: key.substring(10), value: perms.protectedFields[key] };\n        });\n\n      const newProtectedFields: Array<string>[] = [];\n      let overrideProtectedFields = false;\n\n      // check if the object grants the current user access based on the extracted fields\n      protectedFieldsPointerPerm.forEach(pointerPerm => {\n        let pointerPermIncludesUser = false;\n        const readUserFieldValue = object[pointerPerm.key];\n        if (readUserFieldValue) {\n          if (Array.isArray(readUserFieldValue)) {\n            pointerPermIncludesUser = readUserFieldValue.some(\n              user => user.objectId && user.objectId === userId\n            );\n          } else {\n            pointerPermIncludesUser =\n              readUserFieldValue.objectId && readUserFieldValue.objectId === userId;\n          }\n        }\n\n        if (pointerPermIncludesUser) {\n          overrideProtectedFields = true;\n          newProtectedFields.push(pointerPerm.value);\n        }\n      });\n\n      // if at least one pointer-permission affected the current user\n      // intersect vs protectedFields from previous stage (@see addProtectedFields)\n      // Sets theory (intersections): A x (B x C) == (A x B) x C\n      if (overrideProtectedFields && protectedFields) {\n        newProtectedFields.push(protectedFields);\n      }\n      // intersect all sets of protectedFields\n      newProtectedFields.forEach(fields => {\n        if (fields) {\n          // if there're no protctedFields by other criteria ( id / role / auth)\n          // then we must intersect each set (per userField)\n          if (!protectedFields) {\n            protectedFields = fields;\n          } else {\n            protectedFields = protectedFields.filter(v => fields.includes(v));\n          }\n        }\n      });\n    }\n  }\n\n  const isUserClass = className === '_User';\n\n  /* special treat for the user class: don't filter protectedFields if currently loggedin user is\n  the retrieved user */\n  if (!(isUserClass && userId && object.objectId === userId)) {\n    protectedFields && protectedFields.forEach(k => delete object[k]);\n\n    // fields not requested by client (excluded),\n    //but were needed to apply protecttedFields\n    perms.protectedFields &&\n      perms.protectedFields.temporaryKeys &&\n      perms.protectedFields.temporaryKeys.forEach(k => delete object[k]);\n  }\n\n  if (!isUserClass) {\n    return object;\n  }\n\n  object.password = object._hashed_password;\n  delete object._hashed_password;\n\n  delete object.sessionToken;\n\n  if (isMaster) {\n    return object;\n  }\n  delete object._email_verify_token;\n  delete object._perishable_token;\n  delete object._perishable_token_expires_at;\n  delete object._tombstone;\n  delete object._email_verify_token_expires_at;\n  delete object._failed_login_count;\n  delete object._account_lockout_expires_at;\n  delete object._password_changed_at;\n  delete object._password_history;\n\n  if (aclGroup.indexOf(object.objectId) > -1) {\n    return object;\n  }\n  delete object.authData;\n  return object;\n};\n\nimport type { LoadSchemaOptions } from './types';\nimport MongoStorageAdapter from '../Adapters/Storage/Mongo/MongoStorageAdapter';\n\n// Runs an update on the database.\n// Returns a promise for an object with the new values for field\n// modifications that don't know their results ahead of time, like\n// 'increment'.\n// Options:\n//   acl:  a list of strings. If the object to be updated has an ACL,\n//         one of the provided strings must provide the caller with\n//         write permissions.\nconst specialKeysForUpdate = [\n  '_hashed_password',\n  '_perishable_token',\n  '_email_verify_token',\n  '_email_verify_token_expires_at',\n  '_account_lockout_expires_at',\n  '_failed_login_count',\n  '_perishable_token_expires_at',\n  '_password_changed_at',\n  '_password_history',\n];\n\nconst isSpecialUpdateKey = key => {\n  return specialKeysForUpdate.indexOf(key) >= 0;\n};\n\nfunction joinTableName(className, key) {\n  return `_Join:${key}:${className}`;\n}\n\nconst flattenUpdateOperatorsForCreate = object => {\n  for (const key in object) {\n    if (object[key] && object[key].__op) {\n      switch (object[key].__op) {\n        case 'Increment':\n          if (typeof object[key].amount !== 'number') {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'objects to add must be an array');\n          }\n          object[key] = object[key].amount;\n          break;\n        case 'Add':\n          if (!(object[key].objects instanceof Array)) {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'objects to add must be an array');\n          }\n          object[key] = object[key].objects;\n          break;\n        case 'AddUnique':\n          if (!(object[key].objects instanceof Array)) {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'objects to add must be an array');\n          }\n          object[key] = object[key].objects;\n          break;\n        case 'Remove':\n          if (!(object[key].objects instanceof Array)) {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'objects to add must be an array');\n          }\n          object[key] = [];\n          break;\n        case 'Delete':\n          delete object[key];\n          break;\n        default:\n          throw new Parse.Error(\n            Parse.Error.COMMAND_UNAVAILABLE,\n            `The ${object[key].__op} operator is not supported yet.`\n          );\n      }\n    }\n  }\n};\n\nconst transformAuthData = (className, object, schema) => {\n  if (object.authData && className === '_User') {\n    Object.keys(object.authData).forEach(provider => {\n      const providerData = object.authData[provider];\n      const fieldName = `_auth_data_${provider}`;\n      if (providerData == null) {\n        object[fieldName] = {\n          __op: 'Delete',\n        };\n      } else {\n        object[fieldName] = providerData;\n        schema.fields[fieldName] = { type: 'Object' };\n      }\n    });\n    delete object.authData;\n  }\n};\n// Transforms a Database format ACL to a REST API format ACL\nconst untransformObjectACL = ({ _rperm, _wperm, ...output }) => {\n  if (_rperm || _wperm) {\n    output.ACL = {};\n\n    (_rperm || []).forEach(entry => {\n      if (!output.ACL[entry]) {\n        output.ACL[entry] = { read: true };\n      } else {\n        output.ACL[entry]['read'] = true;\n      }\n    });\n\n    (_wperm || []).forEach(entry => {\n      if (!output.ACL[entry]) {\n        output.ACL[entry] = { write: true };\n      } else {\n        output.ACL[entry]['write'] = true;\n      }\n    });\n  }\n  return output;\n};\n\n/**\n * When querying, the fieldName may be compound, extract the root fieldName\n *     `temperature.celsius` becomes `temperature`\n * @param {string} fieldName that may be a compound field name\n * @returns {string} the root name of the field\n */\nconst getRootFieldName = (fieldName: string): string => {\n  return fieldName.split('.')[0];\n};\n\nconst relationSchema = {\n  fields: { relatedId: { type: 'String' }, owningId: { type: 'String' } },\n};\n\nclass DatabaseController {\n  adapter: StorageAdapter;\n  schemaCache: any;\n  schemaPromise: ?Promise<SchemaController.SchemaController>;\n  _transactionalSession: ?any;\n  options: ParseServerOptions;\n\n  constructor(adapter: StorageAdapter, schemaCache: any, options: ParseServerOptions) {\n    this.adapter = adapter;\n    this.schemaCache = schemaCache;\n    // We don't want a mutable this.schema, because then you could have\n    // one request that uses different schemas for different parts of\n    // it. Instead, use loadSchema to get a schema.\n    this.schemaPromise = null;\n    this._transactionalSession = null;\n    this.options = options;\n  }\n\n  collectionExists(className: string): Promise<boolean> {\n    return this.adapter.classExists(className);\n  }\n\n  purgeCollection(className: string): Promise<void> {\n    return this.loadSchema()\n      .then(schemaController => schemaController.getOneSchema(className))\n      .then(schema => this.adapter.deleteObjectsByQuery(className, schema, {}));\n  }\n\n  validateClassName(className: string): Promise<void> {\n    if (!SchemaController.classNameIsValid(className)) {\n      return Promise.reject(\n        new Parse.Error(Parse.Error.INVALID_CLASS_NAME, 'invalid className: ' + className)\n      );\n    }\n    return Promise.resolve();\n  }\n\n  // Returns a promise for a schemaController.\n  loadSchema(\n    options: LoadSchemaOptions = { clearCache: false }\n  ): Promise<SchemaController.SchemaController> {\n    if (this.schemaPromise != null) {\n      return this.schemaPromise;\n    }\n    this.schemaPromise = SchemaController.load(this.adapter, this.schemaCache, options);\n    this.schemaPromise.then(\n      () => delete this.schemaPromise,\n      () => delete this.schemaPromise\n    );\n    return this.loadSchema(options);\n  }\n\n  loadSchemaIfNeeded(\n    schemaController: SchemaController.SchemaController,\n    options: LoadSchemaOptions = { clearCache: false }\n  ): Promise<SchemaController.SchemaController> {\n    return schemaController ? Promise.resolve(schemaController) : this.loadSchema(options);\n  }\n\n  // Returns a promise for the classname that is related to the given\n  // classname through the key.\n  // TODO: make this not in the DatabaseController interface\n  redirectClassNameForKey(className: string, key: string): Promise<?string> {\n    return this.loadSchema().then(schema => {\n      var t = schema.getExpectedType(className, key);\n      if (t != null && typeof t !== 'string' && t.type === 'Relation') {\n        return t.targetClass;\n      }\n      return className;\n    });\n  }\n\n  // Uses the schema to validate the object (REST API format).\n  // Returns a promise that resolves to the new schema.\n  // This does not update this.schema, because in a situation like a\n  // batch request, that could confuse other users of the schema.\n  validateObject(\n    className: string,\n    object: any,\n    query: any,\n    runOptions: QueryOptions\n  ): Promise<boolean> {\n    let schema;\n    const acl = runOptions.acl;\n    const isMaster = acl === undefined;\n    var aclGroup: string[] = acl || [];\n    return this.loadSchema()\n      .then(s => {\n        schema = s;\n        if (isMaster) {\n          return Promise.resolve();\n        }\n        return this.canAddField(schema, className, object, aclGroup, runOptions);\n      })\n      .then(() => {\n        return schema.validateObject(className, object, query);\n      });\n  }\n\n  update(\n    className: string,\n    query: any,\n    update: any,\n    { acl, many, upsert, addsField }: FullQueryOptions = {},\n    skipSanitization: boolean = false,\n    validateOnly: boolean = false,\n    validSchemaController: SchemaController.SchemaController\n  ): Promise<any> {\n    const originalQuery = query;\n    const originalUpdate = update;\n    // Make a copy of the object, so we don't mutate the incoming data.\n    update = deepcopy(update);\n    var relationUpdates = [];\n    var isMaster = acl === undefined;\n    var aclGroup = acl || [];\n\n    return this.loadSchemaIfNeeded(validSchemaController).then(schemaController => {\n      return (isMaster\n        ? Promise.resolve()\n        : schemaController.validatePermission(className, aclGroup, 'update')\n      )\n        .then(() => {\n          relationUpdates = this.collectRelationUpdates(className, originalQuery.objectId, update);\n          if (!isMaster) {\n            query = this.addPointerPermissions(\n              schemaController,\n              className,\n              'update',\n              query,\n              aclGroup\n            );\n\n            if (addsField) {\n              query = {\n                $and: [\n                  query,\n                  this.addPointerPermissions(\n                    schemaController,\n                    className,\n                    'addField',\n                    query,\n                    aclGroup\n                  ),\n                ],\n              };\n            }\n          }\n          if (!query) {\n            return Promise.resolve();\n          }\n          if (acl) {\n            query = addWriteACL(query, acl);\n          }\n          validateQuery(query);\n          return schemaController\n            .getOneSchema(className, true)\n            .catch(error => {\n              // If the schema doesn't exist, pretend it exists with no fields. This behavior\n              // will likely need revisiting.\n              if (error === undefined) {\n                return { fields: {} };\n              }\n              throw error;\n            })\n            .then(schema => {\n              Object.keys(update).forEach(fieldName => {\n                if (fieldName.match(/^authData\\.([a-zA-Z0-9_]+)\\.id$/)) {\n                  throw new Parse.Error(\n                    Parse.Error.INVALID_KEY_NAME,\n                    `Invalid field name for update: ${fieldName}`\n                  );\n                }\n                const rootFieldName = getRootFieldName(fieldName);\n                if (\n                  !SchemaController.fieldNameIsValid(rootFieldName, className) &&\n                  !isSpecialUpdateKey(rootFieldName)\n                ) {\n                  throw new Parse.Error(\n                    Parse.Error.INVALID_KEY_NAME,\n                    `Invalid field name for update: ${fieldName}`\n                  );\n                }\n              });\n              for (const updateOperation in update) {\n                if (\n                  update[updateOperation] &&\n                  typeof update[updateOperation] === 'object' &&\n                  Object.keys(update[updateOperation]).some(\n                    innerKey => innerKey.includes('$') || innerKey.includes('.')\n                  )\n                ) {\n                  throw new Parse.Error(\n                    Parse.Error.INVALID_NESTED_KEY,\n                    \"Nested keys should not contain the '$' or '.' characters\"\n                  );\n                }\n              }\n              update = transformObjectACL(update);\n              transformAuthData(className, update, schema);\n              if (validateOnly) {\n                return this.adapter.find(className, schema, query, {}).then(result => {\n                  if (!result || !result.length) {\n                    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n                  }\n                  return {};\n                });\n              }\n              if (many) {\n                return this.adapter.updateObjectsByQuery(\n                  className,\n                  schema,\n                  query,\n                  update,\n                  this._transactionalSession\n                );\n              } else if (upsert) {\n                return this.adapter.upsertOneObject(\n                  className,\n                  schema,\n                  query,\n                  update,\n                  this._transactionalSession\n                );\n              } else {\n                return this.adapter.findOneAndUpdate(\n                  className,\n                  schema,\n                  query,\n                  update,\n                  this._transactionalSession\n                );\n              }\n            });\n        })\n        .then((result: any) => {\n          if (!result) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n          }\n          if (validateOnly) {\n            return result;\n          }\n          return this.handleRelationUpdates(\n            className,\n            originalQuery.objectId,\n            update,\n            relationUpdates\n          ).then(() => {\n            return result;\n          });\n        })\n        .then(result => {\n          if (skipSanitization) {\n            return Promise.resolve(result);\n          }\n          return this._sanitizeDatabaseResult(originalUpdate, result);\n        });\n    });\n  }\n\n  // Collect all relation-updating operations from a REST-format update.\n  // Returns a list of all relation updates to perform\n  // This mutates update.\n  collectRelationUpdates(className: string, objectId: ?string, update: any) {\n    var ops = [];\n    var deleteMe = [];\n    objectId = update.objectId || objectId;\n\n    var process = (op, key) => {\n      if (!op) {\n        return;\n      }\n      if (op.__op == 'AddRelation') {\n        ops.push({ key, op });\n        deleteMe.push(key);\n      }\n\n      if (op.__op == 'RemoveRelation') {\n        ops.push({ key, op });\n        deleteMe.push(key);\n      }\n\n      if (op.__op == 'Batch') {\n        for (var x of op.ops) {\n          process(x, key);\n        }\n      }\n    };\n\n    for (const key in update) {\n      process(update[key], key);\n    }\n    for (const key of deleteMe) {\n      delete update[key];\n    }\n    return ops;\n  }\n\n  // Processes relation-updating operations from a REST-format update.\n  // Returns a promise that resolves when all updates have been performed\n  handleRelationUpdates(className: string, objectId: string, update: any, ops: any) {\n    var pending = [];\n    objectId = update.objectId || objectId;\n    ops.forEach(({ key, op }) => {\n      if (!op) {\n        return;\n      }\n      if (op.__op == 'AddRelation') {\n        for (const object of op.objects) {\n          pending.push(this.addRelation(key, className, objectId, object.objectId));\n        }\n      }\n\n      if (op.__op == 'RemoveRelation') {\n        for (const object of op.objects) {\n          pending.push(this.removeRelation(key, className, objectId, object.objectId));\n        }\n      }\n    });\n\n    return Promise.all(pending);\n  }\n\n  // Adds a relation.\n  // Returns a promise that resolves successfully iff the add was successful.\n  addRelation(key: string, fromClassName: string, fromId: string, toId: string) {\n    const doc = {\n      relatedId: toId,\n      owningId: fromId,\n    };\n    return this.adapter.upsertOneObject(\n      `_Join:${key}:${fromClassName}`,\n      relationSchema,\n      doc,\n      doc,\n      this._transactionalSession\n    );\n  }\n\n  // Removes a relation.\n  // Returns a promise that resolves successfully iff the remove was\n  // successful.\n  removeRelation(key: string, fromClassName: string, fromId: string, toId: string) {\n    var doc = {\n      relatedId: toId,\n      owningId: fromId,\n    };\n    return this.adapter\n      .deleteObjectsByQuery(\n        `_Join:${key}:${fromClassName}`,\n        relationSchema,\n        doc,\n        this._transactionalSession\n      )\n      .catch(error => {\n        // We don't care if they try to delete a non-existent relation.\n        if (error.code == Parse.Error.OBJECT_NOT_FOUND) {\n          return;\n        }\n        throw error;\n      });\n  }\n\n  // Removes objects matches this query from the database.\n  // Returns a promise that resolves successfully iff the object was\n  // deleted.\n  // Options:\n  //   acl:  a list of strings. If the object to be updated has an ACL,\n  //         one of the provided strings must provide the caller with\n  //         write permissions.\n  destroy(\n    className: string,\n    query: any,\n    { acl }: QueryOptions = {},\n    validSchemaController: SchemaController.SchemaController\n  ): Promise<any> {\n    const isMaster = acl === undefined;\n    const aclGroup = acl || [];\n\n    return this.loadSchemaIfNeeded(validSchemaController).then(schemaController => {\n      return (isMaster\n        ? Promise.resolve()\n        : schemaController.validatePermission(className, aclGroup, 'delete')\n      ).then(() => {\n        if (!isMaster) {\n          query = this.addPointerPermissions(\n            schemaController,\n            className,\n            'delete',\n            query,\n            aclGroup\n          );\n          if (!query) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n          }\n        }\n        // delete by query\n        if (acl) {\n          query = addWriteACL(query, acl);\n        }\n        validateQuery(query);\n        return schemaController\n          .getOneSchema(className)\n          .catch(error => {\n            // If the schema doesn't exist, pretend it exists with no fields. This behavior\n            // will likely need revisiting.\n            if (error === undefined) {\n              return { fields: {} };\n            }\n            throw error;\n          })\n          .then(parseFormatSchema =>\n            this.adapter.deleteObjectsByQuery(\n              className,\n              parseFormatSchema,\n              query,\n              this._transactionalSession\n            )\n          )\n          .catch(error => {\n            // When deleting sessions while changing passwords, don't throw an error if they don't have any sessions.\n            if (className === '_Session' && error.code === Parse.Error.OBJECT_NOT_FOUND) {\n              return Promise.resolve({});\n            }\n            throw error;\n          });\n      });\n    });\n  }\n\n  // Inserts an object into the database.\n  // Returns a promise that resolves successfully iff the object saved.\n  create(\n    className: string,\n    object: any,\n    { acl }: QueryOptions = {},\n    validateOnly: boolean = false,\n    validSchemaController: SchemaController.SchemaController\n  ): Promise<any> {\n    // Make a copy of the object, so we don't mutate the incoming data.\n    const originalObject = object;\n    object = transformObjectACL(object);\n\n    object.createdAt = { iso: object.createdAt, __type: 'Date' };\n    object.updatedAt = { iso: object.updatedAt, __type: 'Date' };\n\n    var isMaster = acl === undefined;\n    var aclGroup = acl || [];\n    const relationUpdates = this.collectRelationUpdates(className, null, object);\n\n    return this.validateClassName(className)\n      .then(() => this.loadSchemaIfNeeded(validSchemaController))\n      .then(schemaController => {\n        return (isMaster\n          ? Promise.resolve()\n          : schemaController.validatePermission(className, aclGroup, 'create')\n        )\n          .then(() => schemaController.enforceClassExists(className))\n          .then(() => schemaController.getOneSchema(className, true))\n          .then(schema => {\n            transformAuthData(className, object, schema);\n            flattenUpdateOperatorsForCreate(object);\n            if (validateOnly) {\n              return {};\n            }\n            return this.adapter.createObject(\n              className,\n              SchemaController.convertSchemaToAdapterSchema(schema),\n              object,\n              this._transactionalSession\n            );\n          })\n          .then(result => {\n            if (validateOnly) {\n              return originalObject;\n            }\n            return this.handleRelationUpdates(\n              className,\n              object.objectId,\n              object,\n              relationUpdates\n            ).then(() => {\n              return this._sanitizeDatabaseResult(originalObject, result.ops[0]);\n            });\n          });\n      });\n  }\n\n  canAddField(\n    schema: SchemaController.SchemaController,\n    className: string,\n    object: any,\n    aclGroup: string[],\n    runOptions: QueryOptions\n  ): Promise<void> {\n    const classSchema = schema.schemaData[className];\n    if (!classSchema) {\n      return Promise.resolve();\n    }\n    const fields = Object.keys(object);\n    const schemaFields = Object.keys(classSchema.fields);\n    const newKeys = fields.filter(field => {\n      // Skip fields that are unset\n      if (object[field] && object[field].__op && object[field].__op === 'Delete') {\n        return false;\n      }\n      return schemaFields.indexOf(field) < 0;\n    });\n    if (newKeys.length > 0) {\n      // adds a marker that new field is being adding during update\n      runOptions.addsField = true;\n\n      const action = runOptions.action;\n      return schema.validatePermission(className, aclGroup, 'addField', action);\n    }\n    return Promise.resolve();\n  }\n\n  // Won't delete collections in the system namespace\n  /**\n   * Delete all classes and clears the schema cache\n   *\n   * @param {boolean} fast set to true if it's ok to just delete rows and not indexes\n   * @returns {Promise<void>} when the deletions completes\n   */\n  deleteEverything(fast: boolean = false): Promise<any> {\n    this.schemaPromise = null;\n    return Promise.all([this.adapter.deleteAllClasses(fast), this.schemaCache.clear()]);\n  }\n\n  // Returns a promise for a list of related ids given an owning id.\n  // className here is the owning className.\n  relatedIds(\n    className: string,\n    key: string,\n    owningId: string,\n    queryOptions: QueryOptions\n  ): Promise<Array<string>> {\n    const { skip, limit, sort } = queryOptions;\n    const findOptions = {};\n    if (sort && sort.createdAt && this.adapter.canSortOnJoinTables) {\n      findOptions.sort = { _id: sort.createdAt };\n      findOptions.limit = limit;\n      findOptions.skip = skip;\n      queryOptions.skip = 0;\n    }\n    return this.adapter\n      .find(joinTableName(className, key), relationSchema, { owningId }, findOptions)\n      .then(results => results.map(result => result.relatedId));\n  }\n\n  // Returns a promise for a list of owning ids given some related ids.\n  // className here is the owning className.\n  owningIds(className: string, key: string, relatedIds: string[]): Promise<string[]> {\n    return this.adapter\n      .find(\n        joinTableName(className, key),\n        relationSchema,\n        { relatedId: { $in: relatedIds } },\n        { keys: ['owningId'] }\n      )\n      .then(results => results.map(result => result.owningId));\n  }\n\n  // Modifies query so that it no longer has $in on relation fields, or\n  // equal-to-pointer constraints on relation fields.\n  // Returns a promise that resolves when query is mutated\n  reduceInRelation(className: string, query: any, schema: any): Promise<any> {\n    // Search for an in-relation or equal-to-relation\n    // Make it sequential for now, not sure of paralleization side effects\n    if (query['$or']) {\n      const ors = query['$or'];\n      return Promise.all(\n        ors.map((aQuery, index) => {\n          return this.reduceInRelation(className, aQuery, schema).then(aQuery => {\n            query['$or'][index] = aQuery;\n          });\n        })\n      ).then(() => {\n        return Promise.resolve(query);\n      });\n    }\n\n    const promises = Object.keys(query).map(key => {\n      const t = schema.getExpectedType(className, key);\n      if (!t || t.type !== 'Relation') {\n        return Promise.resolve(query);\n      }\n      let queries: ?(any[]) = null;\n      if (\n        query[key] &&\n        (query[key]['$in'] ||\n          query[key]['$ne'] ||\n          query[key]['$nin'] ||\n          query[key].__type == 'Pointer')\n      ) {\n        // Build the list of queries\n        queries = Object.keys(query[key]).map(constraintKey => {\n          let relatedIds;\n          let isNegation = false;\n          if (constraintKey === 'objectId') {\n            relatedIds = [query[key].objectId];\n          } else if (constraintKey == '$in') {\n            relatedIds = query[key]['$in'].map(r => r.objectId);\n          } else if (constraintKey == '$nin') {\n            isNegation = true;\n            relatedIds = query[key]['$nin'].map(r => r.objectId);\n          } else if (constraintKey == '$ne') {\n            isNegation = true;\n            relatedIds = [query[key]['$ne'].objectId];\n          } else {\n            return;\n          }\n          return {\n            isNegation,\n            relatedIds,\n          };\n        });\n      } else {\n        queries = [{ isNegation: false, relatedIds: [] }];\n      }\n\n      // remove the current queryKey as we don,t need it anymore\n      delete query[key];\n      // execute each query independently to build the list of\n      // $in / $nin\n      const promises = queries.map(q => {\n        if (!q) {\n          return Promise.resolve();\n        }\n        return this.owningIds(className, key, q.relatedIds).then(ids => {\n          if (q.isNegation) {\n            this.addNotInObjectIdsIds(ids, query);\n          } else {\n            this.addInObjectIdsIds(ids, query);\n          }\n          return Promise.resolve();\n        });\n      });\n\n      return Promise.all(promises).then(() => {\n        return Promise.resolve();\n      });\n    });\n\n    return Promise.all(promises).then(() => {\n      return Promise.resolve(query);\n    });\n  }\n\n  // Modifies query so that it no longer has $relatedTo\n  // Returns a promise that resolves when query is mutated\n  reduceRelationKeys(className: string, query: any, queryOptions: any): ?Promise<void> {\n    if (query['$or']) {\n      return Promise.all(\n        query['$or'].map(aQuery => {\n          return this.reduceRelationKeys(className, aQuery, queryOptions);\n        })\n      );\n    }\n\n    var relatedTo = query['$relatedTo'];\n    if (relatedTo) {\n      return this.relatedIds(\n        relatedTo.object.className,\n        relatedTo.key,\n        relatedTo.object.objectId,\n        queryOptions\n      )\n        .then(ids => {\n          delete query['$relatedTo'];\n          this.addInObjectIdsIds(ids, query);\n          return this.reduceRelationKeys(className, query, queryOptions);\n        })\n        .then(() => {});\n    }\n  }\n\n  addInObjectIdsIds(ids: ?Array<string> = null, query: any) {\n    const idsFromString: ?Array<string> =\n      typeof query.objectId === 'string' ? [query.objectId] : null;\n    const idsFromEq: ?Array<string> =\n      query.objectId && query.objectId['$eq'] ? [query.objectId['$eq']] : null;\n    const idsFromIn: ?Array<string> =\n      query.objectId && query.objectId['$in'] ? query.objectId['$in'] : null;\n\n    // @flow-disable-next\n    const allIds: Array<Array<string>> = [idsFromString, idsFromEq, idsFromIn, ids].filter(\n      list => list !== null\n    );\n    const totalLength = allIds.reduce((memo, list) => memo + list.length, 0);\n\n    let idsIntersection = [];\n    if (totalLength > 125) {\n      idsIntersection = intersect.big(allIds);\n    } else {\n      idsIntersection = intersect(allIds);\n    }\n\n    // Need to make sure we don't clobber existing shorthand $eq constraints on objectId.\n    if (!('objectId' in query)) {\n      query.objectId = {\n        $in: undefined,\n      };\n    } else if (typeof query.objectId === 'string') {\n      query.objectId = {\n        $in: undefined,\n        $eq: query.objectId,\n      };\n    }\n    query.objectId['$in'] = idsIntersection;\n\n    return query;\n  }\n\n  addNotInObjectIdsIds(ids: string[] = [], query: any) {\n    const idsFromNin = query.objectId && query.objectId['$nin'] ? query.objectId['$nin'] : [];\n    let allIds = [...idsFromNin, ...ids].filter(list => list !== null);\n\n    // make a set and spread to remove duplicates\n    allIds = [...new Set(allIds)];\n\n    // Need to make sure we don't clobber existing shorthand $eq constraints on objectId.\n    if (!('objectId' in query)) {\n      query.objectId = {\n        $nin: undefined,\n      };\n    } else if (typeof query.objectId === 'string') {\n      query.objectId = {\n        $nin: undefined,\n        $eq: query.objectId,\n      };\n    }\n\n    query.objectId['$nin'] = allIds;\n    return query;\n  }\n\n  // Runs a query on the database.\n  // Returns a promise that resolves to a list of items.\n  // Options:\n  //   skip    number of results to skip.\n  //   limit   limit to this number of results.\n  //   sort    an object where keys are the fields to sort by.\n  //           the value is +1 for ascending, -1 for descending.\n  //   count   run a count instead of returning results.\n  //   acl     restrict this operation with an ACL for the provided array\n  //           of user objectIds and roles. acl: null means no user.\n  //           when this field is not present, don't do anything regarding ACLs.\n  //  caseInsensitive make string comparisons case insensitive\n  // TODO: make userIds not needed here. The db adapter shouldn't know\n  // anything about users, ideally. Then, improve the format of the ACL\n  // arg to work like the others.\n  find(\n    className: string,\n    query: any,\n    {\n      skip,\n      limit,\n      acl,\n      sort = {},\n      count,\n      keys,\n      op,\n      distinct,\n      pipeline,\n      readPreference,\n      hint,\n      caseInsensitive = false,\n      explain,\n    }: any = {},\n    auth: any = {},\n    validSchemaController: SchemaController.SchemaController\n  ): Promise<any> {\n    const isMaster = acl === undefined;\n    const aclGroup = acl || [];\n    op =\n      op || (typeof query.objectId == 'string' && Object.keys(query).length === 1 ? 'get' : 'find');\n    // Count operation if counting\n    op = count === true ? 'count' : op;\n\n    let classExists = true;\n    return this.loadSchemaIfNeeded(validSchemaController).then(schemaController => {\n      //Allow volatile classes if querying with Master (for _PushStatus)\n      //TODO: Move volatile classes concept into mongo adapter, postgres adapter shouldn't care\n      //that api.parse.com breaks when _PushStatus exists in mongo.\n      return schemaController\n        .getOneSchema(className, isMaster)\n        .catch(error => {\n          // Behavior for non-existent classes is kinda weird on Parse.com. Probably doesn't matter too much.\n          // For now, pretend the class exists but has no objects,\n          if (error === undefined) {\n            classExists = false;\n            return { fields: {} };\n          }\n          throw error;\n        })\n        .then(schema => {\n          // Parse.com treats queries on _created_at and _updated_at as if they were queries on createdAt and updatedAt,\n          // so duplicate that behavior here. If both are specified, the correct behavior to match Parse.com is to\n          // use the one that appears first in the sort list.\n          if (sort._created_at) {\n            sort.createdAt = sort._created_at;\n            delete sort._created_at;\n          }\n          if (sort._updated_at) {\n            sort.updatedAt = sort._updated_at;\n            delete sort._updated_at;\n          }\n          const queryOptions = {\n            skip,\n            limit,\n            sort,\n            keys,\n            readPreference,\n            hint,\n            caseInsensitive,\n            explain,\n          };\n          Object.keys(sort).forEach(fieldName => {\n            if (fieldName.match(/^authData\\.([a-zA-Z0-9_]+)\\.id$/)) {\n              throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Cannot sort by ${fieldName}`);\n            }\n            const rootFieldName = getRootFieldName(fieldName);\n            if (!SchemaController.fieldNameIsValid(rootFieldName, className)) {\n              throw new Parse.Error(\n                Parse.Error.INVALID_KEY_NAME,\n                `Invalid field name: ${fieldName}.`\n              );\n            }\n          });\n          return (isMaster\n            ? Promise.resolve()\n            : schemaController.validatePermission(className, aclGroup, op)\n          )\n            .then(() => this.reduceRelationKeys(className, query, queryOptions))\n            .then(() => this.reduceInRelation(className, query, schemaController))\n            .then(() => {\n              let protectedFields;\n              if (!isMaster) {\n                query = this.addPointerPermissions(\n                  schemaController,\n                  className,\n                  op,\n                  query,\n                  aclGroup\n                );\n                /* Don't use projections to optimize the protectedFields since the protectedFields\n                  based on pointer-permissions are determined after querying. The filtering can\n                  overwrite the protected fields. */\n                protectedFields = this.addProtectedFields(\n                  schemaController,\n                  className,\n                  query,\n                  aclGroup,\n                  auth,\n                  queryOptions\n                );\n              }\n              if (!query) {\n                if (op === 'get') {\n                  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n                } else {\n                  return [];\n                }\n              }\n              if (!isMaster) {\n                if (op === 'update' || op === 'delete') {\n                  query = addWriteACL(query, aclGroup);\n                } else {\n                  query = addReadACL(query, aclGroup);\n                }\n              }\n              validateQuery(query);\n              if (count) {\n                if (!classExists) {\n                  return 0;\n                } else {\n                  return this.adapter.count(\n                    className,\n                    schema,\n                    query,\n                    readPreference,\n                    undefined,\n                    hint\n                  );\n                }\n              } else if (distinct) {\n                if (!classExists) {\n                  return [];\n                } else {\n                  return this.adapter.distinct(className, schema, query, distinct);\n                }\n              } else if (pipeline) {\n                if (!classExists) {\n                  return [];\n                } else {\n                  return this.adapter.aggregate(\n                    className,\n                    schema,\n                    pipeline,\n                    readPreference,\n                    hint,\n                    explain\n                  );\n                }\n              } else if (explain) {\n                return this.adapter.find(className, schema, query, queryOptions);\n              } else {\n                return this.adapter\n                  .find(className, schema, query, queryOptions)\n                  .then(objects =>\n                    objects.map(object => {\n                      object = untransformObjectACL(object);\n                      return filterSensitiveData(\n                        isMaster,\n                        aclGroup,\n                        auth,\n                        op,\n                        schemaController,\n                        className,\n                        protectedFields,\n                        object\n                      );\n                    })\n                  )\n                  .catch(error => {\n                    throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, error);\n                  });\n              }\n            });\n        });\n    });\n  }\n\n  deleteSchema(className: string): Promise<void> {\n    return this.loadSchema({ clearCache: true })\n      .then(schemaController => schemaController.getOneSchema(className, true))\n      .catch(error => {\n        if (error === undefined) {\n          return { fields: {} };\n        } else {\n          throw error;\n        }\n      })\n      .then((schema: any) => {\n        return this.collectionExists(className)\n          .then(() => this.adapter.count(className, { fields: {} }, null, '', false))\n          .then(count => {\n            if (count > 0) {\n              throw new Parse.Error(\n                255,\n                `Class ${className} is not empty, contains ${count} objects, cannot drop schema.`\n              );\n            }\n            return this.adapter.deleteClass(className);\n          })\n          .then(wasParseCollection => {\n            if (wasParseCollection) {\n              const relationFieldNames = Object.keys(schema.fields).filter(\n                fieldName => schema.fields[fieldName].type === 'Relation'\n              );\n              return Promise.all(\n                relationFieldNames.map(name =>\n                  this.adapter.deleteClass(joinTableName(className, name))\n                )\n              ).then(() => {\n                return;\n              });\n            } else {\n              return Promise.resolve();\n            }\n          });\n      });\n  }\n\n  // Constraints query using CLP's pointer permissions (PP) if any.\n  // 1. Etract the user id from caller's ACLgroup;\n  // 2. Exctract a list of field names that are PP for target collection and operation;\n  // 3. Constraint the original query so that each PP field must\n  // point to caller's id (or contain it in case of PP field being an array)\n  addPointerPermissions(\n    schema: SchemaController.SchemaController,\n    className: string,\n    operation: string,\n    query: any,\n    aclGroup: any[] = []\n  ): any {\n    // Check if class has public permission for operation\n    // If the BaseCLP pass, let go through\n    if (schema.testPermissionsForClassName(className, aclGroup, operation)) {\n      return query;\n    }\n    const perms = schema.getClassLevelPermissions(className);\n\n    const userACL = aclGroup.filter(acl => {\n      return acl.indexOf('role:') != 0 && acl != '*';\n    });\n\n    const groupKey =\n      ['get', 'find', 'count'].indexOf(operation) > -1 ? 'readUserFields' : 'writeUserFields';\n\n    const permFields = [];\n\n    if (perms[operation] && perms[operation].pointerFields) {\n      permFields.push(...perms[operation].pointerFields);\n    }\n\n    if (perms[groupKey]) {\n      for (const field of perms[groupKey]) {\n        if (!permFields.includes(field)) {\n          permFields.push(field);\n        }\n      }\n    }\n    // the ACL should have exactly 1 user\n    if (permFields.length > 0) {\n      // the ACL should have exactly 1 user\n      // No user set return undefined\n      // If the length is > 1, that means we didn't de-dupe users correctly\n      if (userACL.length != 1) {\n        return;\n      }\n      const userId = userACL[0];\n      const userPointer = {\n        __type: 'Pointer',\n        className: '_User',\n        objectId: userId,\n      };\n\n      const queries = permFields.map(key => {\n        const fieldDescriptor = schema.getExpectedType(className, key);\n        const fieldType =\n          fieldDescriptor &&\n          typeof fieldDescriptor === 'object' &&\n          Object.prototype.hasOwnProperty.call(fieldDescriptor, 'type')\n            ? fieldDescriptor.type\n            : null;\n\n        let queryClause;\n\n        if (fieldType === 'Pointer') {\n          // constraint for single pointer setup\n          queryClause = { [key]: userPointer };\n        } else if (fieldType === 'Array') {\n          // constraint for users-array setup\n          queryClause = { [key]: { $all: [userPointer] } };\n        } else if (fieldType === 'Object') {\n          // constraint for object setup\n          queryClause = { [key]: userPointer };\n        } else {\n          // This means that there is a CLP field of an unexpected type. This condition should not happen, which is\n          // why is being treated as an error.\n          throw Error(\n            `An unexpected condition occurred when resolving pointer permissions: ${className} ${key}`\n          );\n        }\n        // if we already have a constraint on the key, use the $and\n        if (Object.prototype.hasOwnProperty.call(query, key)) {\n          return { $and: [queryClause, query] };\n        }\n        // otherwise just add the constaint\n        return Object.assign({}, query, queryClause);\n      });\n\n      return queries.length === 1 ? queries[0] : { $or: queries };\n    } else {\n      return query;\n    }\n  }\n\n  addProtectedFields(\n    schema: SchemaController.SchemaController | any,\n    className: string,\n    query: any = {},\n    aclGroup: any[] = [],\n    auth: any = {},\n    queryOptions: FullQueryOptions = {}\n  ): null | string[] {\n    const perms =\n      schema && schema.getClassLevelPermissions\n        ? schema.getClassLevelPermissions(className)\n        : schema;\n    if (!perms) return null;\n\n    const protectedFields = perms.protectedFields;\n    if (!protectedFields) return null;\n\n    if (aclGroup.indexOf(query.objectId) > -1) return null;\n\n    // for queries where \"keys\" are set and do not include all 'userField':{field},\n    // we have to transparently include it, and then remove before returning to client\n    // Because if such key not projected the permission won't be enforced properly\n    // PS this is called when 'excludeKeys' already reduced to 'keys'\n    const preserveKeys = queryOptions.keys;\n\n    // these are keys that need to be included only\n    // to be able to apply protectedFields by pointer\n    // and then unset before returning to client (later in  filterSensitiveFields)\n    const serverOnlyKeys = [];\n\n    const authenticated = auth.user;\n\n    // map to allow check without array search\n    const roles = (auth.userRoles || []).reduce((acc, r) => {\n      acc[r] = protectedFields[r];\n      return acc;\n    }, {});\n\n    // array of sets of protected fields. separate item for each applicable criteria\n    const protectedKeysSets = [];\n\n    for (const key in protectedFields) {\n      // skip userFields\n      if (key.startsWith('userField:')) {\n        if (preserveKeys) {\n          const fieldName = key.substring(10);\n          if (!preserveKeys.includes(fieldName)) {\n            // 1. put it there temporarily\n            queryOptions.keys && queryOptions.keys.push(fieldName);\n            // 2. preserve it delete later\n            serverOnlyKeys.push(fieldName);\n          }\n        }\n        continue;\n      }\n\n      // add public tier\n      if (key === '*') {\n        protectedKeysSets.push(protectedFields[key]);\n        continue;\n      }\n\n      if (authenticated) {\n        if (key === 'authenticated') {\n          // for logged in users\n          protectedKeysSets.push(protectedFields[key]);\n          continue;\n        }\n\n        if (roles[key] && key.startsWith('role:')) {\n          // add applicable roles\n          protectedKeysSets.push(roles[key]);\n        }\n      }\n    }\n\n    // check if there's a rule for current user's id\n    if (authenticated) {\n      const userId = auth.user.id;\n      if (perms.protectedFields[userId]) {\n        protectedKeysSets.push(perms.protectedFields[userId]);\n      }\n    }\n\n    // preserve fields to be removed before sending response to client\n    if (serverOnlyKeys.length > 0) {\n      perms.protectedFields.temporaryKeys = serverOnlyKeys;\n    }\n\n    let protectedKeys = protectedKeysSets.reduce((acc, next) => {\n      if (next) {\n        acc.push(...next);\n      }\n      return acc;\n    }, []);\n\n    // intersect all sets of protectedFields\n    protectedKeysSets.forEach(fields => {\n      if (fields) {\n        protectedKeys = protectedKeys.filter(v => fields.includes(v));\n      }\n    });\n\n    return protectedKeys;\n  }\n\n  createTransactionalSession() {\n    return this.adapter.createTransactionalSession().then(transactionalSession => {\n      this._transactionalSession = transactionalSession;\n    });\n  }\n\n  commitTransactionalSession() {\n    if (!this._transactionalSession) {\n      throw new Error('There is no transactional session to commit');\n    }\n    return this.adapter.commitTransactionalSession(this._transactionalSession).then(() => {\n      this._transactionalSession = null;\n    });\n  }\n\n  abortTransactionalSession() {\n    if (!this._transactionalSession) {\n      throw new Error('There is no transactional session to abort');\n    }\n    return this.adapter.abortTransactionalSession(this._transactionalSession).then(() => {\n      this._transactionalSession = null;\n    });\n  }\n\n  // TODO: create indexes on first creation of a _User object. Otherwise it's impossible to\n  // have a Parse app without it having a _User collection.\n  performInitialization() {\n    const requiredUserFields = {\n      fields: {\n        ...SchemaController.defaultColumns._Default,\n        ...SchemaController.defaultColumns._User,\n      },\n    };\n    const requiredRoleFields = {\n      fields: {\n        ...SchemaController.defaultColumns._Default,\n        ...SchemaController.defaultColumns._Role,\n      },\n    };\n    const requiredIdempotencyFields = {\n      fields: {\n        ...SchemaController.defaultColumns._Default,\n        ...SchemaController.defaultColumns._Idempotency,\n      },\n    };\n\n    const userClassPromise = this.loadSchema().then(schema => schema.enforceClassExists('_User'));\n    const roleClassPromise = this.loadSchema().then(schema => schema.enforceClassExists('_Role'));\n    const idempotencyClassPromise =\n      this.adapter instanceof MongoStorageAdapter\n        ? this.loadSchema().then(schema => schema.enforceClassExists('_Idempotency'))\n        : Promise.resolve();\n\n    const usernameUniqueness = userClassPromise\n      .then(() => this.adapter.ensureUniqueness('_User', requiredUserFields, ['username']))\n      .catch(error => {\n        logger.warn('Unable to ensure uniqueness for usernames: ', error);\n        throw error;\n      });\n\n    const usernameCaseInsensitiveIndex = userClassPromise\n      .then(() =>\n        this.adapter.ensureIndex(\n          '_User',\n          requiredUserFields,\n          ['username'],\n          'case_insensitive_username',\n          true\n        )\n      )\n      .catch(error => {\n        logger.warn('Unable to create case insensitive username index: ', error);\n        throw error;\n      });\n\n    const emailUniqueness = userClassPromise\n      .then(() => this.adapter.ensureUniqueness('_User', requiredUserFields, ['email']))\n      .catch(error => {\n        logger.warn('Unable to ensure uniqueness for user email addresses: ', error);\n        throw error;\n      });\n\n    const emailCaseInsensitiveIndex = userClassPromise\n      .then(() =>\n        this.adapter.ensureIndex(\n          '_User',\n          requiredUserFields,\n          ['email'],\n          'case_insensitive_email',\n          true\n        )\n      )\n      .catch(error => {\n        logger.warn('Unable to create case insensitive email index: ', error);\n        throw error;\n      });\n\n    const roleUniqueness = roleClassPromise\n      .then(() => this.adapter.ensureUniqueness('_Role', requiredRoleFields, ['name']))\n      .catch(error => {\n        logger.warn('Unable to ensure uniqueness for role name: ', error);\n        throw error;\n      });\n\n    const idempotencyRequestIdIndex =\n      this.adapter instanceof MongoStorageAdapter\n        ? idempotencyClassPromise\n          .then(() =>\n            this.adapter.ensureUniqueness('_Idempotency', requiredIdempotencyFields, ['reqId'])\n          )\n          .catch(error => {\n            logger.warn('Unable to ensure uniqueness for idempotency request ID: ', error);\n            throw error;\n          })\n        : Promise.resolve();\n\n    const idempotencyExpireIndex =\n      this.adapter instanceof MongoStorageAdapter\n        ? idempotencyClassPromise\n          .then(() =>\n            this.adapter.ensureIndex(\n              '_Idempotency',\n              requiredIdempotencyFields,\n              ['expire'],\n              'ttl',\n              false,\n              { ttl: 0 }\n            )\n          )\n          .catch(error => {\n            logger.warn('Unable to create TTL index for idempotency expire date: ', error);\n            throw error;\n          })\n        : Promise.resolve();\n\n    const indexPromise = this.adapter.updateSchemaWithIndexes();\n\n    // Create tables for volatile classes\n    const adapterInit = this.adapter.performInitialization({\n      VolatileClassesSchemas: SchemaController.VolatileClassesSchemas,\n    });\n    return Promise.all([\n      usernameUniqueness,\n      usernameCaseInsensitiveIndex,\n      emailUniqueness,\n      emailCaseInsensitiveIndex,\n      roleUniqueness,\n      idempotencyRequestIdIndex,\n      idempotencyExpireIndex,\n      adapterInit,\n      indexPromise,\n    ]);\n  }\n\n  _expandResultOnKeyPath(object: any, key: string, value: any): any {\n    if (key.indexOf('.') < 0) {\n      object[key] = value[key];\n      return object;\n    }\n    const path = key.split('.');\n    const firstKey = path[0];\n    const nextPath = path.slice(1).join('.');\n\n    // Scan request data for denied keywords\n    if (this.options && this.options.requestKeywordDenylist) {\n      // Scan request data for denied keywords\n      for (const keyword of this.options.requestKeywordDenylist) {\n        const match = Utils.objectContainsKeyValue({ firstKey: undefined }, keyword.key, undefined);\n        if (match) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_KEY_NAME,\n            `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`\n          );\n        }\n      }\n    }\n\n    object[firstKey] = this._expandResultOnKeyPath(\n      object[firstKey] || {},\n      nextPath,\n      value[firstKey]\n    );\n    delete object[key];\n    return object;\n  }\n\n  _sanitizeDatabaseResult(originalObject: any, result: any): Promise<any> {\n    const response = {};\n    if (!result) {\n      return Promise.resolve(response);\n    }\n    Object.keys(originalObject).forEach(key => {\n      const keyUpdate = originalObject[key];\n      // determine if that was an op\n      if (\n        keyUpdate &&\n        typeof keyUpdate === 'object' &&\n        keyUpdate.__op &&\n        ['Add', 'AddUnique', 'Remove', 'Increment'].indexOf(keyUpdate.__op) > -1\n      ) {\n        // only valid ops that produce an actionable result\n        // the op may have happened on a keypath\n        this._expandResultOnKeyPath(response, key, result);\n      }\n    });\n    return Promise.resolve(response);\n  }\n\n  static _validateQuery: any => void;\n  static filterSensitiveData: (boolean, any[], any, any, any, string, any[], any) => void;\n}\n\nmodule.exports = DatabaseController;\n// Expose validateQuery for tests\nmodule.exports._validateQuery = validateQuery;\nmodule.exports.filterSensitiveData = filterSensitiveData;\n"]}
|
|
1498
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Controllers/DatabaseController.js"],"names":["addWriteACL","query","acl","newQuery","_","cloneDeep","_wperm","$in","addReadACL","_rperm","transformObjectACL","ACL","result","entry","read","push","write","specialQueryKeys","specialMasterQueryKeys","validateQuery","isMaster","update","Parse","Error","INVALID_QUERY","$or","Array","forEach","value","$and","$nor","length","Object","keys","key","$regex","$options","match","includes","INVALID_KEY_NAME","filterSensitiveData","aclGroup","auth","operation","schema","className","protectedFields","object","userId","user","id","perms","getClassLevelPermissions","isReadOperation","indexOf","protectedFieldsPointerPerm","filter","startsWith","map","substring","newProtectedFields","overrideProtectedFields","pointerPerm","pointerPermIncludesUser","readUserFieldValue","isArray","some","objectId","fields","v","isUserClass","k","temporaryKeys","password","_hashed_password","sessionToken","charAt","authData","specialKeysForUpdate","isSpecialUpdateKey","joinTableName","flattenUpdateOperatorsForCreate","__op","amount","INVALID_JSON","objects","COMMAND_UNAVAILABLE","transformAuthData","provider","providerData","fieldName","type","untransformObjectACL","output","getRootFieldName","split","relationSchema","relatedId","owningId","DatabaseController","constructor","adapter","schemaCache","options","schemaPromise","_transactionalSession","collectionExists","classExists","purgeCollection","loadSchema","then","schemaController","getOneSchema","deleteObjectsByQuery","validateClassName","SchemaController","classNameIsValid","Promise","reject","INVALID_CLASS_NAME","resolve","clearCache","load","loadSchemaIfNeeded","redirectClassNameForKey","t","getExpectedType","targetClass","validateObject","runOptions","undefined","s","canAddField","many","upsert","addsField","skipSanitization","validateOnly","validSchemaController","originalQuery","originalUpdate","relationUpdates","validatePermission","collectRelationUpdates","addPointerPermissions","catch","error","rootFieldName","fieldNameIsValid","updateOperation","innerKey","INVALID_NESTED_KEY","find","OBJECT_NOT_FOUND","updateObjectsByQuery","upsertOneObject","findOneAndUpdate","handleRelationUpdates","_sanitizeDatabaseResult","ops","deleteMe","process","op","x","pending","addRelation","removeRelation","all","fromClassName","fromId","toId","doc","code","destroy","parseFormatSchema","create","originalObject","createdAt","iso","__type","updatedAt","enforceClassExists","createObject","convertSchemaToAdapterSchema","classSchema","schemaData","schemaFields","newKeys","field","action","deleteEverything","fast","deleteAllClasses","clear","relatedIds","queryOptions","skip","limit","sort","findOptions","canSortOnJoinTables","_id","results","owningIds","reduceInRelation","ors","aQuery","index","promises","queries","constraintKey","isNegation","r","q","ids","addNotInObjectIdsIds","addInObjectIdsIds","reduceRelationKeys","relatedTo","idsFromString","idsFromEq","idsFromIn","allIds","list","totalLength","reduce","memo","idsIntersection","intersect","big","$eq","idsFromNin","Set","$nin","count","distinct","pipeline","readPreference","hint","caseInsensitive","explain","_created_at","_updated_at","addProtectedFields","aggregate","INTERNAL_SERVER_ERROR","deleteSchema","deleteClass","wasParseCollection","relationFieldNames","name","testPermissionsForClassName","userACL","groupKey","permFields","pointerFields","userPointer","fieldDescriptor","fieldType","prototype","hasOwnProperty","call","queryClause","$all","assign","preserveKeys","serverOnlyKeys","authenticated","roles","userRoles","acc","protectedKeysSets","protectedKeys","next","createTransactionalSession","transactionalSession","commitTransactionalSession","abortTransactionalSession","performInitialization","requiredUserFields","defaultColumns","_Default","_User","requiredRoleFields","_Role","requiredIdempotencyFields","_Idempotency","userClassPromise","roleClassPromise","idempotencyClassPromise","MongoStorageAdapter","usernameUniqueness","ensureUniqueness","logger","warn","usernameCaseInsensitiveIndex","ensureIndex","emailUniqueness","emailCaseInsensitiveIndex","roleUniqueness","idempotencyRequestIdIndex","idempotencyExpireIndex","ttl","indexPromise","updateSchemaWithIndexes","adapterInit","VolatileClassesSchemas","_expandResultOnKeyPath","path","firstKey","nextPath","slice","join","requestKeywordDenylist","keyword","Utils","objectContainsKeyValue","JSON","stringify","response","keyUpdate","module","exports","_validateQuery"],"mappings":"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","sourcesContent":["﻿// @flow\n// A database adapter that works with data exported from the hosted\n// Parse database.\n\n// @flow-disable-next\nimport { Parse } from 'parse/node';\n// @flow-disable-next\nimport _ from 'lodash';\n// @flow-disable-next\nimport intersect from 'intersect';\n// @flow-disable-next\nimport deepcopy from 'deepcopy';\nimport logger from '../logger';\nimport Utils from '../Utils';\nimport * as SchemaController from './SchemaController';\nimport { StorageAdapter } from '../Adapters/Storage/StorageAdapter';\nimport type { ParseServerOptions } from '../Options';\nimport type { QueryOptions, FullQueryOptions } from '../Adapters/Storage/StorageAdapter';\n\nfunction addWriteACL(query, acl) {\n  const newQuery = _.cloneDeep(query);\n  //Can't be any existing '_wperm' query, we don't allow client queries on that, no need to $and\n  newQuery._wperm = { $in: [null, ...acl] };\n  return newQuery;\n}\n\nfunction addReadACL(query, acl) {\n  const newQuery = _.cloneDeep(query);\n  //Can't be any existing '_rperm' query, we don't allow client queries on that, no need to $and\n  newQuery._rperm = { $in: [null, '*', ...acl] };\n  return newQuery;\n}\n\n// Transforms a REST API formatted ACL object to our two-field mongo format.\nconst transformObjectACL = ({ ACL, ...result }) => {\n  if (!ACL) {\n    return result;\n  }\n\n  result._wperm = [];\n  result._rperm = [];\n\n  for (const entry in ACL) {\n    if (ACL[entry].read) {\n      result._rperm.push(entry);\n    }\n    if (ACL[entry].write) {\n      result._wperm.push(entry);\n    }\n  }\n  return result;\n};\n\nconst specialQueryKeys = ['$and', '$or', '$nor', '_rperm', '_wperm'];\nconst specialMasterQueryKeys = [\n  ...specialQueryKeys,\n  '_email_verify_token',\n  '_perishable_token',\n  '_tombstone',\n  '_email_verify_token_expires_at',\n  '_failed_login_count',\n  '_account_lockout_expires_at',\n  '_password_changed_at',\n  '_password_history',\n];\n\nconst validateQuery = (query: any, isMaster: boolean, update: boolean): void => {\n  if (query.ACL) {\n    throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Cannot query on ACL.');\n  }\n\n  if (query.$or) {\n    if (query.$or instanceof Array) {\n      query.$or.forEach(value => validateQuery(value, isMaster, update));\n    } else {\n      throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Bad $or format - use an array value.');\n    }\n  }\n\n  if (query.$and) {\n    if (query.$and instanceof Array) {\n      query.$and.forEach(value => validateQuery(value, isMaster, update));\n    } else {\n      throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Bad $and format - use an array value.');\n    }\n  }\n\n  if (query.$nor) {\n    if (query.$nor instanceof Array && query.$nor.length > 0) {\n      query.$nor.forEach(value => validateQuery(value, isMaster, update));\n    } else {\n      throw new Parse.Error(\n        Parse.Error.INVALID_QUERY,\n        'Bad $nor format - use an array of at least 1 value.'\n      );\n    }\n  }\n\n  Object.keys(query).forEach(key => {\n    if (query && query[key] && query[key].$regex) {\n      if (typeof query[key].$options === 'string') {\n        if (!query[key].$options.match(/^[imxs]+$/)) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_QUERY,\n            `Bad $options value for query: ${query[key].$options}`\n          );\n        }\n      }\n    }\n    if (\n      !key.match(/^[a-zA-Z][a-zA-Z0-9_\\.]*$/) &&\n      ((!specialQueryKeys.includes(key) && !isMaster && !update) ||\n        (update && isMaster && !specialMasterQueryKeys.includes(key)))\n    ) {\n      throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Invalid key name: ${key}`);\n    }\n  });\n};\n\n// Filters out any data that shouldn't be on this REST-formatted object.\nconst filterSensitiveData = (\n  isMaster: boolean,\n  aclGroup: any[],\n  auth: any,\n  operation: any,\n  schema: SchemaController.SchemaController | any,\n  className: string,\n  protectedFields: null | Array<any>,\n  object: any\n) => {\n  let userId = null;\n  if (auth && auth.user) userId = auth.user.id;\n\n  // replace protectedFields when using pointer-permissions\n  const perms =\n    schema && schema.getClassLevelPermissions ? schema.getClassLevelPermissions(className) : {};\n  if (perms) {\n    const isReadOperation = ['get', 'find'].indexOf(operation) > -1;\n\n    if (isReadOperation && perms.protectedFields) {\n      // extract protectedFields added with the pointer-permission prefix\n      const protectedFieldsPointerPerm = Object.keys(perms.protectedFields)\n        .filter(key => key.startsWith('userField:'))\n        .map(key => {\n          return { key: key.substring(10), value: perms.protectedFields[key] };\n        });\n\n      const newProtectedFields: Array<string>[] = [];\n      let overrideProtectedFields = false;\n\n      // check if the object grants the current user access based on the extracted fields\n      protectedFieldsPointerPerm.forEach(pointerPerm => {\n        let pointerPermIncludesUser = false;\n        const readUserFieldValue = object[pointerPerm.key];\n        if (readUserFieldValue) {\n          if (Array.isArray(readUserFieldValue)) {\n            pointerPermIncludesUser = readUserFieldValue.some(\n              user => user.objectId && user.objectId === userId\n            );\n          } else {\n            pointerPermIncludesUser =\n              readUserFieldValue.objectId && readUserFieldValue.objectId === userId;\n          }\n        }\n\n        if (pointerPermIncludesUser) {\n          overrideProtectedFields = true;\n          newProtectedFields.push(pointerPerm.value);\n        }\n      });\n\n      // if at least one pointer-permission affected the current user\n      // intersect vs protectedFields from previous stage (@see addProtectedFields)\n      // Sets theory (intersections): A x (B x C) == (A x B) x C\n      if (overrideProtectedFields && protectedFields) {\n        newProtectedFields.push(protectedFields);\n      }\n      // intersect all sets of protectedFields\n      newProtectedFields.forEach(fields => {\n        if (fields) {\n          // if there're no protctedFields by other criteria ( id / role / auth)\n          // then we must intersect each set (per userField)\n          if (!protectedFields) {\n            protectedFields = fields;\n          } else {\n            protectedFields = protectedFields.filter(v => fields.includes(v));\n          }\n        }\n      });\n    }\n  }\n\n  const isUserClass = className === '_User';\n\n  /* special treat for the user class: don't filter protectedFields if currently loggedin user is\n  the retrieved user */\n  if (!(isUserClass && userId && object.objectId === userId)) {\n    protectedFields && protectedFields.forEach(k => delete object[k]);\n\n    // fields not requested by client (excluded),\n    //but were needed to apply protecttedFields\n    perms.protectedFields &&\n      perms.protectedFields.temporaryKeys &&\n      perms.protectedFields.temporaryKeys.forEach(k => delete object[k]);\n  }\n\n  if (isUserClass) {\n    object.password = object._hashed_password;\n    delete object._hashed_password;\n    delete object.sessionToken;\n  }\n\n  if (isMaster) {\n    return object;\n  }\n\n  for (const key in object) {\n    if (key.charAt(0) === '_') {\n      delete object[key];\n    }\n  }\n\n  if (!isUserClass) {\n    return object;\n  }\n\n  if (aclGroup.indexOf(object.objectId) > -1) {\n    return object;\n  }\n  delete object.authData;\n  return object;\n};\n\nimport type { LoadSchemaOptions } from './types';\nimport MongoStorageAdapter from '../Adapters/Storage/Mongo/MongoStorageAdapter';\n\n// Runs an update on the database.\n// Returns a promise for an object with the new values for field\n// modifications that don't know their results ahead of time, like\n// 'increment'.\n// Options:\n//   acl:  a list of strings. If the object to be updated has an ACL,\n//         one of the provided strings must provide the caller with\n//         write permissions.\nconst specialKeysForUpdate = [\n  '_hashed_password',\n  '_perishable_token',\n  '_email_verify_token',\n  '_email_verify_token_expires_at',\n  '_account_lockout_expires_at',\n  '_failed_login_count',\n  '_perishable_token_expires_at',\n  '_password_changed_at',\n  '_password_history',\n];\n\nconst isSpecialUpdateKey = key => {\n  return specialKeysForUpdate.indexOf(key) >= 0;\n};\n\nfunction joinTableName(className, key) {\n  return `_Join:${key}:${className}`;\n}\n\nconst flattenUpdateOperatorsForCreate = object => {\n  for (const key in object) {\n    if (object[key] && object[key].__op) {\n      switch (object[key].__op) {\n        case 'Increment':\n          if (typeof object[key].amount !== 'number') {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'objects to add must be an array');\n          }\n          object[key] = object[key].amount;\n          break;\n        case 'Add':\n          if (!(object[key].objects instanceof Array)) {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'objects to add must be an array');\n          }\n          object[key] = object[key].objects;\n          break;\n        case 'AddUnique':\n          if (!(object[key].objects instanceof Array)) {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'objects to add must be an array');\n          }\n          object[key] = object[key].objects;\n          break;\n        case 'Remove':\n          if (!(object[key].objects instanceof Array)) {\n            throw new Parse.Error(Parse.Error.INVALID_JSON, 'objects to add must be an array');\n          }\n          object[key] = [];\n          break;\n        case 'Delete':\n          delete object[key];\n          break;\n        default:\n          throw new Parse.Error(\n            Parse.Error.COMMAND_UNAVAILABLE,\n            `The ${object[key].__op} operator is not supported yet.`\n          );\n      }\n    }\n  }\n};\n\nconst transformAuthData = (className, object, schema) => {\n  if (object.authData && className === '_User') {\n    Object.keys(object.authData).forEach(provider => {\n      const providerData = object.authData[provider];\n      const fieldName = `_auth_data_${provider}`;\n      if (providerData == null) {\n        object[fieldName] = {\n          __op: 'Delete',\n        };\n      } else {\n        object[fieldName] = providerData;\n        schema.fields[fieldName] = { type: 'Object' };\n      }\n    });\n    delete object.authData;\n  }\n};\n// Transforms a Database format ACL to a REST API format ACL\nconst untransformObjectACL = ({ _rperm, _wperm, ...output }) => {\n  if (_rperm || _wperm) {\n    output.ACL = {};\n\n    (_rperm || []).forEach(entry => {\n      if (!output.ACL[entry]) {\n        output.ACL[entry] = { read: true };\n      } else {\n        output.ACL[entry]['read'] = true;\n      }\n    });\n\n    (_wperm || []).forEach(entry => {\n      if (!output.ACL[entry]) {\n        output.ACL[entry] = { write: true };\n      } else {\n        output.ACL[entry]['write'] = true;\n      }\n    });\n  }\n  return output;\n};\n\n/**\n * When querying, the fieldName may be compound, extract the root fieldName\n *     `temperature.celsius` becomes `temperature`\n * @param {string} fieldName that may be a compound field name\n * @returns {string} the root name of the field\n */\nconst getRootFieldName = (fieldName: string): string => {\n  return fieldName.split('.')[0];\n};\n\nconst relationSchema = {\n  fields: { relatedId: { type: 'String' }, owningId: { type: 'String' } },\n};\n\nclass DatabaseController {\n  adapter: StorageAdapter;\n  schemaCache: any;\n  schemaPromise: ?Promise<SchemaController.SchemaController>;\n  _transactionalSession: ?any;\n  options: ParseServerOptions;\n\n  constructor(adapter: StorageAdapter, schemaCache: any, options: ParseServerOptions) {\n    this.adapter = adapter;\n    this.schemaCache = schemaCache;\n    // We don't want a mutable this.schema, because then you could have\n    // one request that uses different schemas for different parts of\n    // it. Instead, use loadSchema to get a schema.\n    this.schemaPromise = null;\n    this._transactionalSession = null;\n    this.options = options;\n  }\n\n  collectionExists(className: string): Promise<boolean> {\n    return this.adapter.classExists(className);\n  }\n\n  purgeCollection(className: string): Promise<void> {\n    return this.loadSchema()\n      .then(schemaController => schemaController.getOneSchema(className))\n      .then(schema => this.adapter.deleteObjectsByQuery(className, schema, {}));\n  }\n\n  validateClassName(className: string): Promise<void> {\n    if (!SchemaController.classNameIsValid(className)) {\n      return Promise.reject(\n        new Parse.Error(Parse.Error.INVALID_CLASS_NAME, 'invalid className: ' + className)\n      );\n    }\n    return Promise.resolve();\n  }\n\n  // Returns a promise for a schemaController.\n  loadSchema(\n    options: LoadSchemaOptions = { clearCache: false }\n  ): Promise<SchemaController.SchemaController> {\n    if (this.schemaPromise != null) {\n      return this.schemaPromise;\n    }\n    this.schemaPromise = SchemaController.load(this.adapter, this.schemaCache, options);\n    this.schemaPromise.then(\n      () => delete this.schemaPromise,\n      () => delete this.schemaPromise\n    );\n    return this.loadSchema(options);\n  }\n\n  loadSchemaIfNeeded(\n    schemaController: SchemaController.SchemaController,\n    options: LoadSchemaOptions = { clearCache: false }\n  ): Promise<SchemaController.SchemaController> {\n    return schemaController ? Promise.resolve(schemaController) : this.loadSchema(options);\n  }\n\n  // Returns a promise for the classname that is related to the given\n  // classname through the key.\n  // TODO: make this not in the DatabaseController interface\n  redirectClassNameForKey(className: string, key: string): Promise<?string> {\n    return this.loadSchema().then(schema => {\n      var t = schema.getExpectedType(className, key);\n      if (t != null && typeof t !== 'string' && t.type === 'Relation') {\n        return t.targetClass;\n      }\n      return className;\n    });\n  }\n\n  // Uses the schema to validate the object (REST API format).\n  // Returns a promise that resolves to the new schema.\n  // This does not update this.schema, because in a situation like a\n  // batch request, that could confuse other users of the schema.\n  validateObject(\n    className: string,\n    object: any,\n    query: any,\n    runOptions: QueryOptions\n  ): Promise<boolean> {\n    let schema;\n    const acl = runOptions.acl;\n    const isMaster = acl === undefined;\n    var aclGroup: string[] = acl || [];\n    return this.loadSchema()\n      .then(s => {\n        schema = s;\n        if (isMaster) {\n          return Promise.resolve();\n        }\n        return this.canAddField(schema, className, object, aclGroup, runOptions);\n      })\n      .then(() => {\n        return schema.validateObject(className, object, query);\n      });\n  }\n\n  update(\n    className: string,\n    query: any,\n    update: any,\n    { acl, many, upsert, addsField }: FullQueryOptions = {},\n    skipSanitization: boolean = false,\n    validateOnly: boolean = false,\n    validSchemaController: SchemaController.SchemaController\n  ): Promise<any> {\n    const originalQuery = query;\n    const originalUpdate = update;\n    // Make a copy of the object, so we don't mutate the incoming data.\n    update = deepcopy(update);\n    var relationUpdates = [];\n    var isMaster = acl === undefined;\n    var aclGroup = acl || [];\n\n    return this.loadSchemaIfNeeded(validSchemaController).then(schemaController => {\n      return (isMaster\n        ? Promise.resolve()\n        : schemaController.validatePermission(className, aclGroup, 'update')\n      )\n        .then(() => {\n          relationUpdates = this.collectRelationUpdates(className, originalQuery.objectId, update);\n          if (!isMaster) {\n            query = this.addPointerPermissions(\n              schemaController,\n              className,\n              'update',\n              query,\n              aclGroup\n            );\n\n            if (addsField) {\n              query = {\n                $and: [\n                  query,\n                  this.addPointerPermissions(\n                    schemaController,\n                    className,\n                    'addField',\n                    query,\n                    aclGroup\n                  ),\n                ],\n              };\n            }\n          }\n          if (!query) {\n            return Promise.resolve();\n          }\n          if (acl) {\n            query = addWriteACL(query, acl);\n          }\n          validateQuery(query, isMaster, true);\n          return schemaController\n            .getOneSchema(className, true)\n            .catch(error => {\n              // If the schema doesn't exist, pretend it exists with no fields. This behavior\n              // will likely need revisiting.\n              if (error === undefined) {\n                return { fields: {} };\n              }\n              throw error;\n            })\n            .then(schema => {\n              Object.keys(update).forEach(fieldName => {\n                if (fieldName.match(/^authData\\.([a-zA-Z0-9_]+)\\.id$/)) {\n                  throw new Parse.Error(\n                    Parse.Error.INVALID_KEY_NAME,\n                    `Invalid field name for update: ${fieldName}`\n                  );\n                }\n                const rootFieldName = getRootFieldName(fieldName);\n                if (\n                  !SchemaController.fieldNameIsValid(rootFieldName, className) &&\n                  !isSpecialUpdateKey(rootFieldName)\n                ) {\n                  throw new Parse.Error(\n                    Parse.Error.INVALID_KEY_NAME,\n                    `Invalid field name for update: ${fieldName}`\n                  );\n                }\n              });\n              for (const updateOperation in update) {\n                if (\n                  update[updateOperation] &&\n                  typeof update[updateOperation] === 'object' &&\n                  Object.keys(update[updateOperation]).some(\n                    innerKey => innerKey.includes('$') || innerKey.includes('.')\n                  )\n                ) {\n                  throw new Parse.Error(\n                    Parse.Error.INVALID_NESTED_KEY,\n                    \"Nested keys should not contain the '$' or '.' characters\"\n                  );\n                }\n              }\n              update = transformObjectACL(update);\n              transformAuthData(className, update, schema);\n              if (validateOnly) {\n                return this.adapter.find(className, schema, query, {}).then(result => {\n                  if (!result || !result.length) {\n                    throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n                  }\n                  return {};\n                });\n              }\n              if (many) {\n                return this.adapter.updateObjectsByQuery(\n                  className,\n                  schema,\n                  query,\n                  update,\n                  this._transactionalSession\n                );\n              } else if (upsert) {\n                return this.adapter.upsertOneObject(\n                  className,\n                  schema,\n                  query,\n                  update,\n                  this._transactionalSession\n                );\n              } else {\n                return this.adapter.findOneAndUpdate(\n                  className,\n                  schema,\n                  query,\n                  update,\n                  this._transactionalSession\n                );\n              }\n            });\n        })\n        .then((result: any) => {\n          if (!result) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n          }\n          if (validateOnly) {\n            return result;\n          }\n          return this.handleRelationUpdates(\n            className,\n            originalQuery.objectId,\n            update,\n            relationUpdates\n          ).then(() => {\n            return result;\n          });\n        })\n        .then(result => {\n          if (skipSanitization) {\n            return Promise.resolve(result);\n          }\n          return this._sanitizeDatabaseResult(originalUpdate, result);\n        });\n    });\n  }\n\n  // Collect all relation-updating operations from a REST-format update.\n  // Returns a list of all relation updates to perform\n  // This mutates update.\n  collectRelationUpdates(className: string, objectId: ?string, update: any) {\n    var ops = [];\n    var deleteMe = [];\n    objectId = update.objectId || objectId;\n\n    var process = (op, key) => {\n      if (!op) {\n        return;\n      }\n      if (op.__op == 'AddRelation') {\n        ops.push({ key, op });\n        deleteMe.push(key);\n      }\n\n      if (op.__op == 'RemoveRelation') {\n        ops.push({ key, op });\n        deleteMe.push(key);\n      }\n\n      if (op.__op == 'Batch') {\n        for (var x of op.ops) {\n          process(x, key);\n        }\n      }\n    };\n\n    for (const key in update) {\n      process(update[key], key);\n    }\n    for (const key of deleteMe) {\n      delete update[key];\n    }\n    return ops;\n  }\n\n  // Processes relation-updating operations from a REST-format update.\n  // Returns a promise that resolves when all updates have been performed\n  handleRelationUpdates(className: string, objectId: string, update: any, ops: any) {\n    var pending = [];\n    objectId = update.objectId || objectId;\n    ops.forEach(({ key, op }) => {\n      if (!op) {\n        return;\n      }\n      if (op.__op == 'AddRelation') {\n        for (const object of op.objects) {\n          pending.push(this.addRelation(key, className, objectId, object.objectId));\n        }\n      }\n\n      if (op.__op == 'RemoveRelation') {\n        for (const object of op.objects) {\n          pending.push(this.removeRelation(key, className, objectId, object.objectId));\n        }\n      }\n    });\n\n    return Promise.all(pending);\n  }\n\n  // Adds a relation.\n  // Returns a promise that resolves successfully iff the add was successful.\n  addRelation(key: string, fromClassName: string, fromId: string, toId: string) {\n    const doc = {\n      relatedId: toId,\n      owningId: fromId,\n    };\n    return this.adapter.upsertOneObject(\n      `_Join:${key}:${fromClassName}`,\n      relationSchema,\n      doc,\n      doc,\n      this._transactionalSession\n    );\n  }\n\n  // Removes a relation.\n  // Returns a promise that resolves successfully iff the remove was\n  // successful.\n  removeRelation(key: string, fromClassName: string, fromId: string, toId: string) {\n    var doc = {\n      relatedId: toId,\n      owningId: fromId,\n    };\n    return this.adapter\n      .deleteObjectsByQuery(\n        `_Join:${key}:${fromClassName}`,\n        relationSchema,\n        doc,\n        this._transactionalSession\n      )\n      .catch(error => {\n        // We don't care if they try to delete a non-existent relation.\n        if (error.code == Parse.Error.OBJECT_NOT_FOUND) {\n          return;\n        }\n        throw error;\n      });\n  }\n\n  // Removes objects matches this query from the database.\n  // Returns a promise that resolves successfully iff the object was\n  // deleted.\n  // Options:\n  //   acl:  a list of strings. If the object to be updated has an ACL,\n  //         one of the provided strings must provide the caller with\n  //         write permissions.\n  destroy(\n    className: string,\n    query: any,\n    { acl }: QueryOptions = {},\n    validSchemaController: SchemaController.SchemaController\n  ): Promise<any> {\n    const isMaster = acl === undefined;\n    const aclGroup = acl || [];\n\n    return this.loadSchemaIfNeeded(validSchemaController).then(schemaController => {\n      return (isMaster\n        ? Promise.resolve()\n        : schemaController.validatePermission(className, aclGroup, 'delete')\n      ).then(() => {\n        if (!isMaster) {\n          query = this.addPointerPermissions(\n            schemaController,\n            className,\n            'delete',\n            query,\n            aclGroup\n          );\n          if (!query) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n          }\n        }\n        // delete by query\n        if (acl) {\n          query = addWriteACL(query, acl);\n        }\n        validateQuery(query, isMaster, false);\n        return schemaController\n          .getOneSchema(className)\n          .catch(error => {\n            // If the schema doesn't exist, pretend it exists with no fields. This behavior\n            // will likely need revisiting.\n            if (error === undefined) {\n              return { fields: {} };\n            }\n            throw error;\n          })\n          .then(parseFormatSchema =>\n            this.adapter.deleteObjectsByQuery(\n              className,\n              parseFormatSchema,\n              query,\n              this._transactionalSession\n            )\n          )\n          .catch(error => {\n            // When deleting sessions while changing passwords, don't throw an error if they don't have any sessions.\n            if (className === '_Session' && error.code === Parse.Error.OBJECT_NOT_FOUND) {\n              return Promise.resolve({});\n            }\n            throw error;\n          });\n      });\n    });\n  }\n\n  // Inserts an object into the database.\n  // Returns a promise that resolves successfully iff the object saved.\n  create(\n    className: string,\n    object: any,\n    { acl }: QueryOptions = {},\n    validateOnly: boolean = false,\n    validSchemaController: SchemaController.SchemaController\n  ): Promise<any> {\n    // Make a copy of the object, so we don't mutate the incoming data.\n    const originalObject = object;\n    object = transformObjectACL(object);\n\n    object.createdAt = { iso: object.createdAt, __type: 'Date' };\n    object.updatedAt = { iso: object.updatedAt, __type: 'Date' };\n\n    var isMaster = acl === undefined;\n    var aclGroup = acl || [];\n    const relationUpdates = this.collectRelationUpdates(className, null, object);\n\n    return this.validateClassName(className)\n      .then(() => this.loadSchemaIfNeeded(validSchemaController))\n      .then(schemaController => {\n        return (isMaster\n          ? Promise.resolve()\n          : schemaController.validatePermission(className, aclGroup, 'create')\n        )\n          .then(() => schemaController.enforceClassExists(className))\n          .then(() => schemaController.getOneSchema(className, true))\n          .then(schema => {\n            transformAuthData(className, object, schema);\n            flattenUpdateOperatorsForCreate(object);\n            if (validateOnly) {\n              return {};\n            }\n            return this.adapter.createObject(\n              className,\n              SchemaController.convertSchemaToAdapterSchema(schema),\n              object,\n              this._transactionalSession\n            );\n          })\n          .then(result => {\n            if (validateOnly) {\n              return originalObject;\n            }\n            return this.handleRelationUpdates(\n              className,\n              object.objectId,\n              object,\n              relationUpdates\n            ).then(() => {\n              return this._sanitizeDatabaseResult(originalObject, result.ops[0]);\n            });\n          });\n      });\n  }\n\n  canAddField(\n    schema: SchemaController.SchemaController,\n    className: string,\n    object: any,\n    aclGroup: string[],\n    runOptions: QueryOptions\n  ): Promise<void> {\n    const classSchema = schema.schemaData[className];\n    if (!classSchema) {\n      return Promise.resolve();\n    }\n    const fields = Object.keys(object);\n    const schemaFields = Object.keys(classSchema.fields);\n    const newKeys = fields.filter(field => {\n      // Skip fields that are unset\n      if (object[field] && object[field].__op && object[field].__op === 'Delete') {\n        return false;\n      }\n      return schemaFields.indexOf(field) < 0;\n    });\n    if (newKeys.length > 0) {\n      // adds a marker that new field is being adding during update\n      runOptions.addsField = true;\n\n      const action = runOptions.action;\n      return schema.validatePermission(className, aclGroup, 'addField', action);\n    }\n    return Promise.resolve();\n  }\n\n  // Won't delete collections in the system namespace\n  /**\n   * Delete all classes and clears the schema cache\n   *\n   * @param {boolean} fast set to true if it's ok to just delete rows and not indexes\n   * @returns {Promise<void>} when the deletions completes\n   */\n  deleteEverything(fast: boolean = false): Promise<any> {\n    this.schemaPromise = null;\n    return Promise.all([this.adapter.deleteAllClasses(fast), this.schemaCache.clear()]);\n  }\n\n  // Returns a promise for a list of related ids given an owning id.\n  // className here is the owning className.\n  relatedIds(\n    className: string,\n    key: string,\n    owningId: string,\n    queryOptions: QueryOptions\n  ): Promise<Array<string>> {\n    const { skip, limit, sort } = queryOptions;\n    const findOptions = {};\n    if (sort && sort.createdAt && this.adapter.canSortOnJoinTables) {\n      findOptions.sort = { _id: sort.createdAt };\n      findOptions.limit = limit;\n      findOptions.skip = skip;\n      queryOptions.skip = 0;\n    }\n    return this.adapter\n      .find(joinTableName(className, key), relationSchema, { owningId }, findOptions)\n      .then(results => results.map(result => result.relatedId));\n  }\n\n  // Returns a promise for a list of owning ids given some related ids.\n  // className here is the owning className.\n  owningIds(className: string, key: string, relatedIds: string[]): Promise<string[]> {\n    return this.adapter\n      .find(\n        joinTableName(className, key),\n        relationSchema,\n        { relatedId: { $in: relatedIds } },\n        { keys: ['owningId'] }\n      )\n      .then(results => results.map(result => result.owningId));\n  }\n\n  // Modifies query so that it no longer has $in on relation fields, or\n  // equal-to-pointer constraints on relation fields.\n  // Returns a promise that resolves when query is mutated\n  reduceInRelation(className: string, query: any, schema: any): Promise<any> {\n    // Search for an in-relation or equal-to-relation\n    // Make it sequential for now, not sure of paralleization side effects\n    if (query['$or']) {\n      const ors = query['$or'];\n      return Promise.all(\n        ors.map((aQuery, index) => {\n          return this.reduceInRelation(className, aQuery, schema).then(aQuery => {\n            query['$or'][index] = aQuery;\n          });\n        })\n      ).then(() => {\n        return Promise.resolve(query);\n      });\n    }\n\n    const promises = Object.keys(query).map(key => {\n      const t = schema.getExpectedType(className, key);\n      if (!t || t.type !== 'Relation') {\n        return Promise.resolve(query);\n      }\n      let queries: ?(any[]) = null;\n      if (\n        query[key] &&\n        (query[key]['$in'] ||\n          query[key]['$ne'] ||\n          query[key]['$nin'] ||\n          query[key].__type == 'Pointer')\n      ) {\n        // Build the list of queries\n        queries = Object.keys(query[key]).map(constraintKey => {\n          let relatedIds;\n          let isNegation = false;\n          if (constraintKey === 'objectId') {\n            relatedIds = [query[key].objectId];\n          } else if (constraintKey == '$in') {\n            relatedIds = query[key]['$in'].map(r => r.objectId);\n          } else if (constraintKey == '$nin') {\n            isNegation = true;\n            relatedIds = query[key]['$nin'].map(r => r.objectId);\n          } else if (constraintKey == '$ne') {\n            isNegation = true;\n            relatedIds = [query[key]['$ne'].objectId];\n          } else {\n            return;\n          }\n          return {\n            isNegation,\n            relatedIds,\n          };\n        });\n      } else {\n        queries = [{ isNegation: false, relatedIds: [] }];\n      }\n\n      // remove the current queryKey as we don,t need it anymore\n      delete query[key];\n      // execute each query independently to build the list of\n      // $in / $nin\n      const promises = queries.map(q => {\n        if (!q) {\n          return Promise.resolve();\n        }\n        return this.owningIds(className, key, q.relatedIds).then(ids => {\n          if (q.isNegation) {\n            this.addNotInObjectIdsIds(ids, query);\n          } else {\n            this.addInObjectIdsIds(ids, query);\n          }\n          return Promise.resolve();\n        });\n      });\n\n      return Promise.all(promises).then(() => {\n        return Promise.resolve();\n      });\n    });\n\n    return Promise.all(promises).then(() => {\n      return Promise.resolve(query);\n    });\n  }\n\n  // Modifies query so that it no longer has $relatedTo\n  // Returns a promise that resolves when query is mutated\n  reduceRelationKeys(className: string, query: any, queryOptions: any): ?Promise<void> {\n    if (query['$or']) {\n      return Promise.all(\n        query['$or'].map(aQuery => {\n          return this.reduceRelationKeys(className, aQuery, queryOptions);\n        })\n      );\n    }\n\n    var relatedTo = query['$relatedTo'];\n    if (relatedTo) {\n      return this.relatedIds(\n        relatedTo.object.className,\n        relatedTo.key,\n        relatedTo.object.objectId,\n        queryOptions\n      )\n        .then(ids => {\n          delete query['$relatedTo'];\n          this.addInObjectIdsIds(ids, query);\n          return this.reduceRelationKeys(className, query, queryOptions);\n        })\n        .then(() => {});\n    }\n  }\n\n  addInObjectIdsIds(ids: ?Array<string> = null, query: any) {\n    const idsFromString: ?Array<string> =\n      typeof query.objectId === 'string' ? [query.objectId] : null;\n    const idsFromEq: ?Array<string> =\n      query.objectId && query.objectId['$eq'] ? [query.objectId['$eq']] : null;\n    const idsFromIn: ?Array<string> =\n      query.objectId && query.objectId['$in'] ? query.objectId['$in'] : null;\n\n    // @flow-disable-next\n    const allIds: Array<Array<string>> = [idsFromString, idsFromEq, idsFromIn, ids].filter(\n      list => list !== null\n    );\n    const totalLength = allIds.reduce((memo, list) => memo + list.length, 0);\n\n    let idsIntersection = [];\n    if (totalLength > 125) {\n      idsIntersection = intersect.big(allIds);\n    } else {\n      idsIntersection = intersect(allIds);\n    }\n\n    // Need to make sure we don't clobber existing shorthand $eq constraints on objectId.\n    if (!('objectId' in query)) {\n      query.objectId = {\n        $in: undefined,\n      };\n    } else if (typeof query.objectId === 'string') {\n      query.objectId = {\n        $in: undefined,\n        $eq: query.objectId,\n      };\n    }\n    query.objectId['$in'] = idsIntersection;\n\n    return query;\n  }\n\n  addNotInObjectIdsIds(ids: string[] = [], query: any) {\n    const idsFromNin = query.objectId && query.objectId['$nin'] ? query.objectId['$nin'] : [];\n    let allIds = [...idsFromNin, ...ids].filter(list => list !== null);\n\n    // make a set and spread to remove duplicates\n    allIds = [...new Set(allIds)];\n\n    // Need to make sure we don't clobber existing shorthand $eq constraints on objectId.\n    if (!('objectId' in query)) {\n      query.objectId = {\n        $nin: undefined,\n      };\n    } else if (typeof query.objectId === 'string') {\n      query.objectId = {\n        $nin: undefined,\n        $eq: query.objectId,\n      };\n    }\n\n    query.objectId['$nin'] = allIds;\n    return query;\n  }\n\n  // Runs a query on the database.\n  // Returns a promise that resolves to a list of items.\n  // Options:\n  //   skip    number of results to skip.\n  //   limit   limit to this number of results.\n  //   sort    an object where keys are the fields to sort by.\n  //           the value is +1 for ascending, -1 for descending.\n  //   count   run a count instead of returning results.\n  //   acl     restrict this operation with an ACL for the provided array\n  //           of user objectIds and roles. acl: null means no user.\n  //           when this field is not present, don't do anything regarding ACLs.\n  //  caseInsensitive make string comparisons case insensitive\n  // TODO: make userIds not needed here. The db adapter shouldn't know\n  // anything about users, ideally. Then, improve the format of the ACL\n  // arg to work like the others.\n  find(\n    className: string,\n    query: any,\n    {\n      skip,\n      limit,\n      acl,\n      sort = {},\n      count,\n      keys,\n      op,\n      distinct,\n      pipeline,\n      readPreference,\n      hint,\n      caseInsensitive = false,\n      explain,\n    }: any = {},\n    auth: any = {},\n    validSchemaController: SchemaController.SchemaController\n  ): Promise<any> {\n    const isMaster = acl === undefined;\n    const aclGroup = acl || [];\n    op =\n      op || (typeof query.objectId == 'string' && Object.keys(query).length === 1 ? 'get' : 'find');\n    // Count operation if counting\n    op = count === true ? 'count' : op;\n\n    let classExists = true;\n    return this.loadSchemaIfNeeded(validSchemaController).then(schemaController => {\n      //Allow volatile classes if querying with Master (for _PushStatus)\n      //TODO: Move volatile classes concept into mongo adapter, postgres adapter shouldn't care\n      //that api.parse.com breaks when _PushStatus exists in mongo.\n      return schemaController\n        .getOneSchema(className, isMaster)\n        .catch(error => {\n          // Behavior for non-existent classes is kinda weird on Parse.com. Probably doesn't matter too much.\n          // For now, pretend the class exists but has no objects,\n          if (error === undefined) {\n            classExists = false;\n            return { fields: {} };\n          }\n          throw error;\n        })\n        .then(schema => {\n          // Parse.com treats queries on _created_at and _updated_at as if they were queries on createdAt and updatedAt,\n          // so duplicate that behavior here. If both are specified, the correct behavior to match Parse.com is to\n          // use the one that appears first in the sort list.\n          if (sort._created_at) {\n            sort.createdAt = sort._created_at;\n            delete sort._created_at;\n          }\n          if (sort._updated_at) {\n            sort.updatedAt = sort._updated_at;\n            delete sort._updated_at;\n          }\n          const queryOptions = {\n            skip,\n            limit,\n            sort,\n            keys,\n            readPreference,\n            hint,\n            caseInsensitive,\n            explain,\n          };\n          Object.keys(sort).forEach(fieldName => {\n            if (fieldName.match(/^authData\\.([a-zA-Z0-9_]+)\\.id$/)) {\n              throw new Parse.Error(Parse.Error.INVALID_KEY_NAME, `Cannot sort by ${fieldName}`);\n            }\n            const rootFieldName = getRootFieldName(fieldName);\n            if (!SchemaController.fieldNameIsValid(rootFieldName, className)) {\n              throw new Parse.Error(\n                Parse.Error.INVALID_KEY_NAME,\n                `Invalid field name: ${fieldName}.`\n              );\n            }\n          });\n          return (isMaster\n            ? Promise.resolve()\n            : schemaController.validatePermission(className, aclGroup, op)\n          )\n            .then(() => this.reduceRelationKeys(className, query, queryOptions))\n            .then(() => this.reduceInRelation(className, query, schemaController))\n            .then(() => {\n              let protectedFields;\n              if (!isMaster) {\n                query = this.addPointerPermissions(\n                  schemaController,\n                  className,\n                  op,\n                  query,\n                  aclGroup\n                );\n                /* Don't use projections to optimize the protectedFields since the protectedFields\n                  based on pointer-permissions are determined after querying. The filtering can\n                  overwrite the protected fields. */\n                protectedFields = this.addProtectedFields(\n                  schemaController,\n                  className,\n                  query,\n                  aclGroup,\n                  auth,\n                  queryOptions\n                );\n              }\n              if (!query) {\n                if (op === 'get') {\n                  throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n                } else {\n                  return [];\n                }\n              }\n              if (!isMaster) {\n                if (op === 'update' || op === 'delete') {\n                  query = addWriteACL(query, aclGroup);\n                } else {\n                  query = addReadACL(query, aclGroup);\n                }\n              }\n              validateQuery(query, isMaster, false);\n              if (count) {\n                if (!classExists) {\n                  return 0;\n                } else {\n                  return this.adapter.count(\n                    className,\n                    schema,\n                    query,\n                    readPreference,\n                    undefined,\n                    hint\n                  );\n                }\n              } else if (distinct) {\n                if (!classExists) {\n                  return [];\n                } else {\n                  return this.adapter.distinct(className, schema, query, distinct);\n                }\n              } else if (pipeline) {\n                if (!classExists) {\n                  return [];\n                } else {\n                  return this.adapter.aggregate(\n                    className,\n                    schema,\n                    pipeline,\n                    readPreference,\n                    hint,\n                    explain\n                  );\n                }\n              } else if (explain) {\n                return this.adapter.find(className, schema, query, queryOptions);\n              } else {\n                return this.adapter\n                  .find(className, schema, query, queryOptions)\n                  .then(objects =>\n                    objects.map(object => {\n                      object = untransformObjectACL(object);\n                      return filterSensitiveData(\n                        isMaster,\n                        aclGroup,\n                        auth,\n                        op,\n                        schemaController,\n                        className,\n                        protectedFields,\n                        object\n                      );\n                    })\n                  )\n                  .catch(error => {\n                    throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, error);\n                  });\n              }\n            });\n        });\n    });\n  }\n\n  deleteSchema(className: string): Promise<void> {\n    return this.loadSchema({ clearCache: true })\n      .then(schemaController => schemaController.getOneSchema(className, true))\n      .catch(error => {\n        if (error === undefined) {\n          return { fields: {} };\n        } else {\n          throw error;\n        }\n      })\n      .then((schema: any) => {\n        return this.collectionExists(className)\n          .then(() => this.adapter.count(className, { fields: {} }, null, '', false))\n          .then(count => {\n            if (count > 0) {\n              throw new Parse.Error(\n                255,\n                `Class ${className} is not empty, contains ${count} objects, cannot drop schema.`\n              );\n            }\n            return this.adapter.deleteClass(className);\n          })\n          .then(wasParseCollection => {\n            if (wasParseCollection) {\n              const relationFieldNames = Object.keys(schema.fields).filter(\n                fieldName => schema.fields[fieldName].type === 'Relation'\n              );\n              return Promise.all(\n                relationFieldNames.map(name =>\n                  this.adapter.deleteClass(joinTableName(className, name))\n                )\n              ).then(() => {\n                return;\n              });\n            } else {\n              return Promise.resolve();\n            }\n          });\n      });\n  }\n\n  // Constraints query using CLP's pointer permissions (PP) if any.\n  // 1. Etract the user id from caller's ACLgroup;\n  // 2. Exctract a list of field names that are PP for target collection and operation;\n  // 3. Constraint the original query so that each PP field must\n  // point to caller's id (or contain it in case of PP field being an array)\n  addPointerPermissions(\n    schema: SchemaController.SchemaController,\n    className: string,\n    operation: string,\n    query: any,\n    aclGroup: any[] = []\n  ): any {\n    // Check if class has public permission for operation\n    // If the BaseCLP pass, let go through\n    if (schema.testPermissionsForClassName(className, aclGroup, operation)) {\n      return query;\n    }\n    const perms = schema.getClassLevelPermissions(className);\n\n    const userACL = aclGroup.filter(acl => {\n      return acl.indexOf('role:') != 0 && acl != '*';\n    });\n\n    const groupKey =\n      ['get', 'find', 'count'].indexOf(operation) > -1 ? 'readUserFields' : 'writeUserFields';\n\n    const permFields = [];\n\n    if (perms[operation] && perms[operation].pointerFields) {\n      permFields.push(...perms[operation].pointerFields);\n    }\n\n    if (perms[groupKey]) {\n      for (const field of perms[groupKey]) {\n        if (!permFields.includes(field)) {\n          permFields.push(field);\n        }\n      }\n    }\n    // the ACL should have exactly 1 user\n    if (permFields.length > 0) {\n      // the ACL should have exactly 1 user\n      // No user set return undefined\n      // If the length is > 1, that means we didn't de-dupe users correctly\n      if (userACL.length != 1) {\n        return;\n      }\n      const userId = userACL[0];\n      const userPointer = {\n        __type: 'Pointer',\n        className: '_User',\n        objectId: userId,\n      };\n\n      const queries = permFields.map(key => {\n        const fieldDescriptor = schema.getExpectedType(className, key);\n        const fieldType =\n          fieldDescriptor &&\n          typeof fieldDescriptor === 'object' &&\n          Object.prototype.hasOwnProperty.call(fieldDescriptor, 'type')\n            ? fieldDescriptor.type\n            : null;\n\n        let queryClause;\n\n        if (fieldType === 'Pointer') {\n          // constraint for single pointer setup\n          queryClause = { [key]: userPointer };\n        } else if (fieldType === 'Array') {\n          // constraint for users-array setup\n          queryClause = { [key]: { $all: [userPointer] } };\n        } else if (fieldType === 'Object') {\n          // constraint for object setup\n          queryClause = { [key]: userPointer };\n        } else {\n          // This means that there is a CLP field of an unexpected type. This condition should not happen, which is\n          // why is being treated as an error.\n          throw Error(\n            `An unexpected condition occurred when resolving pointer permissions: ${className} ${key}`\n          );\n        }\n        // if we already have a constraint on the key, use the $and\n        if (Object.prototype.hasOwnProperty.call(query, key)) {\n          return { $and: [queryClause, query] };\n        }\n        // otherwise just add the constaint\n        return Object.assign({}, query, queryClause);\n      });\n\n      return queries.length === 1 ? queries[0] : { $or: queries };\n    } else {\n      return query;\n    }\n  }\n\n  addProtectedFields(\n    schema: SchemaController.SchemaController | any,\n    className: string,\n    query: any = {},\n    aclGroup: any[] = [],\n    auth: any = {},\n    queryOptions: FullQueryOptions = {}\n  ): null | string[] {\n    const perms =\n      schema && schema.getClassLevelPermissions\n        ? schema.getClassLevelPermissions(className)\n        : schema;\n    if (!perms) return null;\n\n    const protectedFields = perms.protectedFields;\n    if (!protectedFields) return null;\n\n    if (aclGroup.indexOf(query.objectId) > -1) return null;\n\n    // for queries where \"keys\" are set and do not include all 'userField':{field},\n    // we have to transparently include it, and then remove before returning to client\n    // Because if such key not projected the permission won't be enforced properly\n    // PS this is called when 'excludeKeys' already reduced to 'keys'\n    const preserveKeys = queryOptions.keys;\n\n    // these are keys that need to be included only\n    // to be able to apply protectedFields by pointer\n    // and then unset before returning to client (later in  filterSensitiveFields)\n    const serverOnlyKeys = [];\n\n    const authenticated = auth.user;\n\n    // map to allow check without array search\n    const roles = (auth.userRoles || []).reduce((acc, r) => {\n      acc[r] = protectedFields[r];\n      return acc;\n    }, {});\n\n    // array of sets of protected fields. separate item for each applicable criteria\n    const protectedKeysSets = [];\n\n    for (const key in protectedFields) {\n      // skip userFields\n      if (key.startsWith('userField:')) {\n        if (preserveKeys) {\n          const fieldName = key.substring(10);\n          if (!preserveKeys.includes(fieldName)) {\n            // 1. put it there temporarily\n            queryOptions.keys && queryOptions.keys.push(fieldName);\n            // 2. preserve it delete later\n            serverOnlyKeys.push(fieldName);\n          }\n        }\n        continue;\n      }\n\n      // add public tier\n      if (key === '*') {\n        protectedKeysSets.push(protectedFields[key]);\n        continue;\n      }\n\n      if (authenticated) {\n        if (key === 'authenticated') {\n          // for logged in users\n          protectedKeysSets.push(protectedFields[key]);\n          continue;\n        }\n\n        if (roles[key] && key.startsWith('role:')) {\n          // add applicable roles\n          protectedKeysSets.push(roles[key]);\n        }\n      }\n    }\n\n    // check if there's a rule for current user's id\n    if (authenticated) {\n      const userId = auth.user.id;\n      if (perms.protectedFields[userId]) {\n        protectedKeysSets.push(perms.protectedFields[userId]);\n      }\n    }\n\n    // preserve fields to be removed before sending response to client\n    if (serverOnlyKeys.length > 0) {\n      perms.protectedFields.temporaryKeys = serverOnlyKeys;\n    }\n\n    let protectedKeys = protectedKeysSets.reduce((acc, next) => {\n      if (next) {\n        acc.push(...next);\n      }\n      return acc;\n    }, []);\n\n    // intersect all sets of protectedFields\n    protectedKeysSets.forEach(fields => {\n      if (fields) {\n        protectedKeys = protectedKeys.filter(v => fields.includes(v));\n      }\n    });\n\n    return protectedKeys;\n  }\n\n  createTransactionalSession() {\n    return this.adapter.createTransactionalSession().then(transactionalSession => {\n      this._transactionalSession = transactionalSession;\n    });\n  }\n\n  commitTransactionalSession() {\n    if (!this._transactionalSession) {\n      throw new Error('There is no transactional session to commit');\n    }\n    return this.adapter.commitTransactionalSession(this._transactionalSession).then(() => {\n      this._transactionalSession = null;\n    });\n  }\n\n  abortTransactionalSession() {\n    if (!this._transactionalSession) {\n      throw new Error('There is no transactional session to abort');\n    }\n    return this.adapter.abortTransactionalSession(this._transactionalSession).then(() => {\n      this._transactionalSession = null;\n    });\n  }\n\n  // TODO: create indexes on first creation of a _User object. Otherwise it's impossible to\n  // have a Parse app without it having a _User collection.\n  performInitialization() {\n    const requiredUserFields = {\n      fields: {\n        ...SchemaController.defaultColumns._Default,\n        ...SchemaController.defaultColumns._User,\n      },\n    };\n    const requiredRoleFields = {\n      fields: {\n        ...SchemaController.defaultColumns._Default,\n        ...SchemaController.defaultColumns._Role,\n      },\n    };\n    const requiredIdempotencyFields = {\n      fields: {\n        ...SchemaController.defaultColumns._Default,\n        ...SchemaController.defaultColumns._Idempotency,\n      },\n    };\n\n    const userClassPromise = this.loadSchema().then(schema => schema.enforceClassExists('_User'));\n    const roleClassPromise = this.loadSchema().then(schema => schema.enforceClassExists('_Role'));\n    const idempotencyClassPromise =\n      this.adapter instanceof MongoStorageAdapter\n        ? this.loadSchema().then(schema => schema.enforceClassExists('_Idempotency'))\n        : Promise.resolve();\n\n    const usernameUniqueness = userClassPromise\n      .then(() => this.adapter.ensureUniqueness('_User', requiredUserFields, ['username']))\n      .catch(error => {\n        logger.warn('Unable to ensure uniqueness for usernames: ', error);\n        throw error;\n      });\n\n    const usernameCaseInsensitiveIndex = userClassPromise\n      .then(() =>\n        this.adapter.ensureIndex(\n          '_User',\n          requiredUserFields,\n          ['username'],\n          'case_insensitive_username',\n          true\n        )\n      )\n      .catch(error => {\n        logger.warn('Unable to create case insensitive username index: ', error);\n        throw error;\n      });\n\n    const emailUniqueness = userClassPromise\n      .then(() => this.adapter.ensureUniqueness('_User', requiredUserFields, ['email']))\n      .catch(error => {\n        logger.warn('Unable to ensure uniqueness for user email addresses: ', error);\n        throw error;\n      });\n\n    const emailCaseInsensitiveIndex = userClassPromise\n      .then(() =>\n        this.adapter.ensureIndex(\n          '_User',\n          requiredUserFields,\n          ['email'],\n          'case_insensitive_email',\n          true\n        )\n      )\n      .catch(error => {\n        logger.warn('Unable to create case insensitive email index: ', error);\n        throw error;\n      });\n\n    const roleUniqueness = roleClassPromise\n      .then(() => this.adapter.ensureUniqueness('_Role', requiredRoleFields, ['name']))\n      .catch(error => {\n        logger.warn('Unable to ensure uniqueness for role name: ', error);\n        throw error;\n      });\n\n    const idempotencyRequestIdIndex =\n      this.adapter instanceof MongoStorageAdapter\n        ? idempotencyClassPromise\n          .then(() =>\n            this.adapter.ensureUniqueness('_Idempotency', requiredIdempotencyFields, ['reqId'])\n          )\n          .catch(error => {\n            logger.warn('Unable to ensure uniqueness for idempotency request ID: ', error);\n            throw error;\n          })\n        : Promise.resolve();\n\n    const idempotencyExpireIndex =\n      this.adapter instanceof MongoStorageAdapter\n        ? idempotencyClassPromise\n          .then(() =>\n            this.adapter.ensureIndex(\n              '_Idempotency',\n              requiredIdempotencyFields,\n              ['expire'],\n              'ttl',\n              false,\n              { ttl: 0 }\n            )\n          )\n          .catch(error => {\n            logger.warn('Unable to create TTL index for idempotency expire date: ', error);\n            throw error;\n          })\n        : Promise.resolve();\n\n    const indexPromise = this.adapter.updateSchemaWithIndexes();\n\n    // Create tables for volatile classes\n    const adapterInit = this.adapter.performInitialization({\n      VolatileClassesSchemas: SchemaController.VolatileClassesSchemas,\n    });\n    return Promise.all([\n      usernameUniqueness,\n      usernameCaseInsensitiveIndex,\n      emailUniqueness,\n      emailCaseInsensitiveIndex,\n      roleUniqueness,\n      idempotencyRequestIdIndex,\n      idempotencyExpireIndex,\n      adapterInit,\n      indexPromise,\n    ]);\n  }\n\n  _expandResultOnKeyPath(object: any, key: string, value: any): any {\n    if (key.indexOf('.') < 0) {\n      object[key] = value[key];\n      return object;\n    }\n    const path = key.split('.');\n    const firstKey = path[0];\n    const nextPath = path.slice(1).join('.');\n\n    // Scan request data for denied keywords\n    if (this.options && this.options.requestKeywordDenylist) {\n      // Scan request data for denied keywords\n      for (const keyword of this.options.requestKeywordDenylist) {\n        const match = Utils.objectContainsKeyValue({ firstKey: undefined }, keyword.key, undefined);\n        if (match) {\n          throw new Parse.Error(\n            Parse.Error.INVALID_KEY_NAME,\n            `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`\n          );\n        }\n      }\n    }\n\n    object[firstKey] = this._expandResultOnKeyPath(\n      object[firstKey] || {},\n      nextPath,\n      value[firstKey]\n    );\n    delete object[key];\n    return object;\n  }\n\n  _sanitizeDatabaseResult(originalObject: any, result: any): Promise<any> {\n    const response = {};\n    if (!result) {\n      return Promise.resolve(response);\n    }\n    Object.keys(originalObject).forEach(key => {\n      const keyUpdate = originalObject[key];\n      // determine if that was an op\n      if (\n        keyUpdate &&\n        typeof keyUpdate === 'object' &&\n        keyUpdate.__op &&\n        ['Add', 'AddUnique', 'Remove', 'Increment'].indexOf(keyUpdate.__op) > -1\n      ) {\n        // only valid ops that produce an actionable result\n        // the op may have happened on a keypath\n        this._expandResultOnKeyPath(response, key, result);\n      }\n    });\n    return Promise.resolve(response);\n  }\n\n  static _validateQuery: (any, boolean, boolean) => void;\n  static filterSensitiveData: (boolean, any[], any, any, any, string, any[], any) => void;\n}\n\nmodule.exports = DatabaseController;\n// Expose validateQuery for tests\nmodule.exports._validateQuery = validateQuery;\nmodule.exports.filterSensitiveData = filterSensitiveData;\n"]}
|
package/lib/RestQuery.js
CHANGED
|
@@ -188,6 +188,8 @@ function RestQuery(config, auth, className, restWhere = {}, restOptions = {}, cl
|
|
|
188
188
|
RestQuery.prototype.execute = function (executeOptions) {
|
|
189
189
|
return Promise.resolve().then(() => {
|
|
190
190
|
return this.buildRestWhere();
|
|
191
|
+
}).then(() => {
|
|
192
|
+
return this.denyProtectedFields();
|
|
191
193
|
}).then(() => {
|
|
192
194
|
return this.handleIncludeAll();
|
|
193
195
|
}).then(() => {
|
|
@@ -642,6 +644,21 @@ RestQuery.prototype.runCount = function () {
|
|
|
642
644
|
return this.config.database.find(this.className, this.restWhere, this.findOptions).then(c => {
|
|
643
645
|
this.response.count = c;
|
|
644
646
|
});
|
|
647
|
+
};
|
|
648
|
+
|
|
649
|
+
RestQuery.prototype.denyProtectedFields = async function () {
|
|
650
|
+
if (this.auth.isMaster) {
|
|
651
|
+
return;
|
|
652
|
+
}
|
|
653
|
+
|
|
654
|
+
const schemaController = await this.config.database.loadSchema();
|
|
655
|
+
const protectedFields = this.config.database.addProtectedFields(schemaController, this.className, this.restWhere, this.findOptions.acl, this.auth, this.findOptions) || [];
|
|
656
|
+
|
|
657
|
+
for (const key of protectedFields) {
|
|
658
|
+
if (this.restWhere[key]) {
|
|
659
|
+
throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, `This user is not allowed to query ${key} on class ${this.className}`);
|
|
660
|
+
}
|
|
661
|
+
}
|
|
645
662
|
}; // Augments this.response with all pointers on an object
|
|
646
663
|
|
|
647
664
|
|
|
@@ -975,4 +992,4 @@ function findObjectWithKey(root, key) {
|
|
|
975
992
|
}
|
|
976
993
|
|
|
977
994
|
module.exports = RestQuery;
|
|
978
|
-
//# 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","response","findOptions","isMaster","user","Error","INVALID_SESSION_TOKEN","$and","__type","objectId","id","doCount","includeAll","include","Object","prototype","hasOwnProperty","call","keysForInclude","keys","split","filter","key","length","map","slice","lastIndexOf","join","option","concat","Array","from","Set","exclude","excludeKeys","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","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","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","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) {\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\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\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    const keysForInclude = restOptions.keys\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.split(',').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') {\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.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(config, auth, className, restWhere, restOptions, clientSDK);\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\nconst 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          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\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    )\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.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"]}
|
|
995
|
+
//# 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","response","findOptions","isMaster","user","Error","INVALID_SESSION_TOKEN","$and","__type","objectId","id","doCount","includeAll","include","Object","prototype","hasOwnProperty","call","keysForInclude","keys","split","filter","key","length","map","slice","lastIndexOf","join","option","concat","Array","from","Set","exclude","excludeKeys","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","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) {\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\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\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    const keysForInclude = restOptions.keys\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.split(',').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') {\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(config, auth, className, restWhere, restOptions, clientSDK);\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\nconst 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          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    )\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.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"]}
|