From ed8b34762e6f3d87a28d95347cdde146e9642b91 Mon Sep 17 00:00:00 2001 From: Ricardo Paiva Date: Wed, 26 Sep 2018 11:41:25 -0300 Subject: [PATCH 1/3] Better push performance --- src/Controllers/PushController.js | 13 ++++++++++++- 1 file changed, 12 insertions(+), 1 deletion(-) diff --git a/src/Controllers/PushController.js b/src/Controllers/PushController.js index 8b645a3680..f58fb67ad6 100644 --- a/src/Controllers/PushController.js +++ b/src/Controllers/PushController.js @@ -4,6 +4,7 @@ import RestWrite from '../RestWrite'; import { master } from '../Auth'; import { pushStatusHandler } from '../StatusHandler'; import { applyDeviceTokenExists } from '../Push/utils'; +import { logger } from '../logger'; export class PushController { @@ -58,6 +59,8 @@ export class PushController { badgeUpdate = () => { // Build a real RestQuery so we can use it in RestWrite const restQuery = new RestQuery(config, master(config), '_Installation', updateWhere); + // change $exists for $ne null for better performance + if (restQuery.restWhere && restQuery.restWhere.deviceToken && restQuery.restWhere.deviceToken['$exists']) restQuery.restWhere.deviceToken = {$ne: null} return restQuery.buildRestWhere().then(() => { const write = new RestWrite(config, master(config), '_Installation', restQuery.restWhere, restUpdate); write.runOptions.many = true; @@ -70,7 +73,15 @@ export class PushController { return pushStatus.setInitial(body, where); }).then(() => { onPushStatusSaved(pushStatus.objectId); - return badgeUpdate(); + const promise = badgeUpdate(); + // add this to ignore badge update errors as default + if (!config.stopOnBadgeUpdateError) { + promise.catch(err => { + logger.info(`Badge update error will be ignored for push status ${pushStatus.objectId}`) + logger.error(err) + }) + } + return promise }).then(() => { // Update audience lastUsed and timesUsed if (body.audience_id) { From ee51f58520982b5dcd996494318c0f527bf9ff9b Mon Sep 17 00:00:00 2001 From: Ricardo Paiva Date: Wed, 26 Sep 2018 12:09:08 -0300 Subject: [PATCH 2/3] Better push performance --- lib/Adapters/Storage/Mongo/MongoCollection.js | 2 +- .../Storage/Mongo/MongoStorageAdapter.js | 3 ++- lib/Controllers/PushController.js | 16 ++++++++++++++-- package-lock.json | 14 +++++++------- 4 files changed, 24 insertions(+), 11 deletions(-) diff --git a/lib/Adapters/Storage/Mongo/MongoCollection.js b/lib/Adapters/Storage/Mongo/MongoCollection.js index 2407f49262..d04ad9b414 100644 --- a/lib/Adapters/Storage/Mongo/MongoCollection.js +++ b/lib/Adapters/Storage/Mongo/MongoCollection.js @@ -110,4 +110,4 @@ class MongoCollection { } } exports.default = MongoCollection; -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/Adapters/Storage/Mongo/MongoCollection.js"],"names":["mongodb","require","Collection","MongoCollection","constructor","mongoCollection","_mongoCollection","find","query","skip","limit","sort","keys","maxTimeMS","readPreference","$score","score","$meta","_rawFind","catch","error","code","message","match","key","index","createIndex","then","findOperation","project","toArray","count","countOperation","distinct","field","aggregate","pipeline","insertOne","object","upsertOne","update","upsert","updateOne","updateMany","deleteMany","_ensureSparseUniqueIndexInBackground","indexRequest","Promise","resolve","reject","ensureIndex","unique","background","sparse","drop"],"mappings":"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","file":"MongoCollection.js","sourcesContent":["const mongodb = require('mongodb');\nconst Collection = mongodb.Collection;\n\nexport default class MongoCollection {\n  _mongoCollection:Collection;\n\n  constructor(mongoCollection:Collection) {\n    this._mongoCollection = mongoCollection;\n  }\n\n  // Does a find with \"smart indexing\".\n  // Currently this just means, if it needs a geoindex and there is\n  // none, then build the geoindex.\n  // This could be improved a lot but it's not clear if that's a good\n  // idea. Or even if this behavior is a good idea.\n  find(query, { skip, limit, sort, keys, maxTimeMS, readPreference } = {}) {\n    // Support for Full Text Search - $text\n    if(keys && keys.$score) {\n      delete keys.$score;\n      keys.score = {$meta: 'textScore'};\n    }\n    return this._rawFind(query, { skip, limit, sort, keys, maxTimeMS, readPreference })\n      .catch(error => {\n        // Check for \"no geoindex\" error\n        if (error.code != 17007 && !error.message.match(/unable to find index for .geoNear/)) {\n          throw error;\n        }\n        // Figure out what key needs an index\n        const key = error.message.match(/field=([A-Za-z_0-9]+) /)[1];\n        if (!key) {\n          throw error;\n        }\n\n        var index = {};\n        index[key] = '2d';\n        return this._mongoCollection.createIndex(index)\n          // Retry, but just once.\n          .then(() => this._rawFind(query, { skip, limit, sort, keys, maxTimeMS, readPreference }));\n      });\n  }\n\n  _rawFind(query, { skip, limit, sort, keys, maxTimeMS, readPreference } = {}) {\n    let findOperation = this._mongoCollection\n      .find(query, { skip, limit, sort, readPreference })\n\n    if (keys) {\n      findOperation = findOperation.project(keys);\n    }\n\n    if (maxTimeMS) {\n      findOperation = findOperation.maxTimeMS(maxTimeMS);\n    }\n\n    return findOperation.toArray();\n  }\n\n  count(query, { skip, limit, sort, maxTimeMS, readPreference } = {}) {\n    const countOperation = this._mongoCollection.count(query, { skip, limit, sort, maxTimeMS, readPreference });\n\n    return countOperation;\n  }\n\n  distinct(field, query) {\n    return this._mongoCollection.distinct(field, query);\n  }\n\n  aggregate(pipeline, { maxTimeMS, readPreference } = {}) {\n    return this._mongoCollection.aggregate(pipeline, { maxTimeMS, readPreference }).toArray();\n  }\n\n  insertOne(object) {\n    return this._mongoCollection.insertOne(object);\n  }\n\n  // Atomically updates data in the database for a single (first) object that matched the query\n  // If there is nothing that matches the query - does insert\n  // Postgres Note: `INSERT ... ON CONFLICT UPDATE` that is available since 9.5.\n  upsertOne(query, update) {\n    return this._mongoCollection.update(query, update, { upsert: true })\n  }\n\n  updateOne(query, update) {\n    return this._mongoCollection.updateOne(query, update);\n  }\n\n  updateMany(query, update) {\n    return this._mongoCollection.updateMany(query, update);\n  }\n\n  deleteMany(query) {\n    return this._mongoCollection.deleteMany(query);\n  }\n\n  _ensureSparseUniqueIndexInBackground(indexRequest) {\n    return new Promise((resolve, reject) => {\n      this._mongoCollection.ensureIndex(indexRequest, { unique: true, background: true, sparse: true }, (error) => {\n        if (error) {\n          reject(error);\n        } else {\n          resolve();\n        }\n      });\n    });\n  }\n\n  drop() {\n    return this._mongoCollection.drop();\n  }\n}\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/Adapters/Storage/Mongo/MongoCollection.js"],"names":["mongodb","require","Collection","MongoCollection","constructor","mongoCollection","_mongoCollection","find","query","skip","limit","sort","keys","maxTimeMS","readPreference","$score","score","$meta","_rawFind","catch","error","code","message","match","key","index","createIndex","background","then","findOperation","project","toArray","count","countOperation","distinct","field","aggregate","pipeline","insertOne","object","upsertOne","update","upsert","updateOne","updateMany","deleteMany","_ensureSparseUniqueIndexInBackground","indexRequest","Promise","resolve","reject","ensureIndex","unique","sparse","drop"],"mappings":"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","file":"MongoCollection.js","sourcesContent":["const mongodb = require('mongodb');\nconst Collection = mongodb.Collection;\n\nexport default class MongoCollection {\n  _mongoCollection:Collection;\n\n  constructor(mongoCollection:Collection) {\n    this._mongoCollection = mongoCollection;\n  }\n\n  // Does a find with \"smart indexing\".\n  // Currently this just means, if it needs a geoindex and there is\n  // none, then build the geoindex.\n  // This could be improved a lot but it's not clear if that's a good\n  // idea. Or even if this behavior is a good idea.\n  find(query, { skip, limit, sort, keys, maxTimeMS, readPreference } = {}) {\n    // Support for Full Text Search - $text\n    if(keys && keys.$score) {\n      delete keys.$score;\n      keys.score = {$meta: 'textScore'};\n    }\n    return this._rawFind(query, { skip, limit, sort, keys, maxTimeMS, readPreference })\n      .catch(error => {\n        // Check for \"no geoindex\" error\n        if (error.code != 17007 && !error.message.match(/unable to find index for .geoNear/)) {\n          throw error;\n        }\n        // Figure out what key needs an index\n        const key = error.message.match(/field=([A-Za-z_0-9]+) /)[1];\n        if (!key) {\n          throw error;\n        }\n\n        var index = {};\n        index[key] = '2d';\n        return this._mongoCollection.createIndex(index, {background: true})\n          // Retry, but just once.\n          .then(() => this._rawFind(query, { skip, limit, sort, keys, maxTimeMS, readPreference }));\n      });\n  }\n\n  _rawFind(query, { skip, limit, sort, keys, maxTimeMS, readPreference } = {}) {\n    let findOperation = this._mongoCollection\n      .find(query, { skip, limit, sort, readPreference })\n\n    if (keys) {\n      findOperation = findOperation.project(keys);\n    }\n\n    if (maxTimeMS) {\n      findOperation = findOperation.maxTimeMS(maxTimeMS);\n    }\n\n    return findOperation.toArray();\n  }\n\n  count(query, { skip, limit, sort, maxTimeMS, readPreference } = {}) {\n    const countOperation = this._mongoCollection.count(query, { skip, limit, sort, maxTimeMS, readPreference });\n\n    return countOperation;\n  }\n\n  distinct(field, query) {\n    return this._mongoCollection.distinct(field, query);\n  }\n\n  aggregate(pipeline, { maxTimeMS, readPreference } = {}) {\n    return this._mongoCollection.aggregate(pipeline, { maxTimeMS, readPreference }).toArray();\n  }\n\n  insertOne(object) {\n    return this._mongoCollection.insertOne(object);\n  }\n\n  // Atomically updates data in the database for a single (first) object that matched the query\n  // If there is nothing that matches the query - does insert\n  // Postgres Note: `INSERT ... ON CONFLICT UPDATE` that is available since 9.5.\n  upsertOne(query, update) {\n    return this._mongoCollection.update(query, update, { upsert: true })\n  }\n\n  updateOne(query, update) {\n    return this._mongoCollection.updateOne(query, update);\n  }\n\n  updateMany(query, update) {\n    return this._mongoCollection.updateMany(query, update);\n  }\n\n  deleteMany(query) {\n    return this._mongoCollection.deleteMany(query);\n  }\n\n  _ensureSparseUniqueIndexInBackground(indexRequest) {\n    return new Promise((resolve, reject) => {\n      this._mongoCollection.ensureIndex(indexRequest, { unique: true, background: true, sparse: true }, (error) => {\n        if (error) {\n          reject(error);\n        } else {\n          resolve();\n        }\n      });\n    });\n  }\n\n  drop() {\n    return this._mongoCollection.drop();\n  }\n}\n"]} \ No newline at end of file diff --git a/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js b/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js index dbd8421b3c..a1a20c35c1 100644 --- a/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +++ b/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js @@ -127,6 +127,7 @@ class MongoStorageAdapter { this._uri = uri; this._collectionPrefix = collectionPrefix; this._mongoOptions = mongoOptions; + this._mongoOptions.useNewUrlParser = true; // MaxTimeMS is not a global MongoDB client option, it is applied per operation. this._maxTimeMS = mongoOptions.maxTimeMS; @@ -769,4 +770,4 @@ class MongoStorageAdapter { exports.MongoStorageAdapter = MongoStorageAdapter; exports.default = MongoStorageAdapter; -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/Adapters/Storage/Mongo/MongoStorageAdapter.js"],"names":["mongodb","require","MongoClient","ReadPreference","MongoSchemaCollectionName","storageAdapterAllCollections","mongoAdapter","connect","then","database","collections","filter","collection","namespace","match","collectionName","indexOf","_collectionPrefix","convertParseSchemaToMongoSchema","schema","fields","_rperm","_wperm","className","_hashed_password","mongoSchemaFromFieldsAndClassNameAndCLP","classLevelPermissions","indexes","mongoObject","_id","objectId","updatedAt","createdAt","_metadata","undefined","fieldName","MongoSchemaCollection","parseFieldTypeToMongoFieldType","class_permissions","Object","keys","length","MongoStorageAdapter","constructor","uri","defaults","DefaultMongoURI","collectionPrefix","mongoOptions","_uri","_mongoOptions","_maxTimeMS","maxTimeMS","canSortOnJoinTables","connectionPromise","encodedUri","client","options","s","db","dbName","on","catch","err","Promise","reject","handleError","error","code","logger","handleShutdown","close","_adaptiveCollection","name","rawCollection","MongoCollection","_schemaCollection","classExists","listCollections","toArray","setClassLevelPermissions","CLPs","schemaCollection","updateSchema","$set","setIndexesWithSchemaFormat","submittedIndexes","existingIndexes","resolve","_id_","deletePromises","insertedIndexes","forEach","field","__op","Parse","Error","INVALID_QUERY","promise","dropIndex","push","key","hasOwnProperty","insertPromise","createIndexes","all","setIndexesFromMongo","getIndexes","reduce","obj","index","_fts","_ftsx","weights","createClass","insertSchema","addFieldIfNotExists","type","createIndexesIfNeeded","deleteClass","drop","message","findAndDeleteSchema","deleteAllClasses","fast","map","remove","deleteFields","fieldNames","mongoFormatNames","collectionUpdate","schemaUpdate","updateMany","getAllClasses","schemasCollection","_fetchAllSchemasFrom_SCHEMA","getClass","_fetchOneSchemaFrom_SCHEMA","createObject","object","insertOne","DUPLICATE_VALUE","underlyingError","matches","Array","isArray","userInfo","duplicated_field","deleteObjectsByQuery","query","mongoWhere","deleteMany","result","n","OBJECT_NOT_FOUND","INTERNAL_SERVER_ERROR","updateObjectsByQuery","update","mongoUpdate","findOneAndUpdate","_mongoCollection","findAndModify","new","value","upsertOneObject","upsertOne","find","skip","limit","sort","readPreference","mongoSort","_","mapKeys","mongoKeys","memo","_parseReadPreference","createTextIndexesIfNeeded","objects","ensureUniqueness","indexCreationRequest","mongoFieldNames","_ensureSparseUniqueIndexInBackground","_rawFind","count","distinct","isPointerField","substring","aggregate","pipeline","stage","$group","_parseAggregateGroupArgs","$match","_parseAggregateArgs","$project","_parseAggregateProjectArgs","results","split","isEmpty","returnValue","targetClass","_convertToDate","Date","PRIMARY","PRIMARY_PREFERRED","SECONDARY","SECONDARY_PREFERRED","NEAREST","performInitialization","createIndex","background","$text","indexName","textIndex","dropAllIndexes","dropIndexes","updateSchemaWithIndexes","classes","promises"],"mappings":"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","file":"MongoStorageAdapter.js","sourcesContent":["// @flow\nimport MongoCollection       from './MongoCollection';\nimport MongoSchemaCollection from './MongoSchemaCollection';\nimport { StorageAdapter }    from '../StorageAdapter';\nimport type { SchemaType,\n  QueryType,\n  StorageClass,\n  QueryOptions } from '../StorageAdapter';\nimport {\n  parse as parseUrl,\n  format as formatUrl,\n} from '../../../vendor/mongodbUrl';\nimport {\n  parseObjectToMongoObjectForCreate,\n  mongoObjectToParseObject,\n  transformKey,\n  transformWhere,\n  transformUpdate,\n  transformPointerString,\n} from './MongoTransform';\n// @flow-disable-next\nimport Parse                 from 'parse/node';\n// @flow-disable-next\nimport _                     from 'lodash';\nimport defaults              from '../../../defaults';\nimport logger                from '../../../logger';\n\n// @flow-disable-next\nconst mongodb = require('mongodb');\nconst MongoClient = mongodb.MongoClient;\nconst ReadPreference = mongodb.ReadPreference;\n\nconst MongoSchemaCollectionName = '_SCHEMA';\n\nconst storageAdapterAllCollections = mongoAdapter => {\n  return mongoAdapter.connect()\n    .then(() => mongoAdapter.database.collections())\n    .then(collections => {\n      return collections.filter(collection => {\n        if (collection.namespace.match(/\\.system\\./)) {\n          return false;\n        }\n        // TODO: If you have one app with a collection prefix that happens to be a prefix of another\n        // apps prefix, this will go very very badly. We should fix that somehow.\n        return (collection.collectionName.indexOf(mongoAdapter._collectionPrefix) == 0);\n      });\n    });\n}\n\nconst convertParseSchemaToMongoSchema = ({...schema}) => {\n  delete schema.fields._rperm;\n  delete schema.fields._wperm;\n\n  if (schema.className === '_User') {\n    // Legacy mongo adapter knows about the difference between password and _hashed_password.\n    // Future database adapters will only know about _hashed_password.\n    // Note: Parse Server will bring back password with injectDefaultSchema, so we don't need\n    // to add _hashed_password back ever.\n    delete schema.fields._hashed_password;\n  }\n\n  return schema;\n}\n\n// Returns { code, error } if invalid, or { result }, an object\n// suitable for inserting into _SCHEMA collection, otherwise.\nconst mongoSchemaFromFieldsAndClassNameAndCLP = (fields, className, classLevelPermissions, indexes) => {\n  const mongoObject = {\n    _id: className,\n    objectId: 'string',\n    updatedAt: 'string',\n    createdAt: 'string',\n    _metadata: undefined,\n  };\n\n  for (const fieldName in fields) {\n    mongoObject[fieldName] = MongoSchemaCollection.parseFieldTypeToMongoFieldType(fields[fieldName]);\n  }\n\n  if (typeof classLevelPermissions !== 'undefined') {\n    mongoObject._metadata = mongoObject._metadata || {};\n    if (!classLevelPermissions) {\n      delete mongoObject._metadata.class_permissions;\n    } else {\n      mongoObject._metadata.class_permissions = classLevelPermissions;\n    }\n  }\n\n  if (indexes && typeof indexes === 'object' && Object.keys(indexes).length > 0) {\n    mongoObject._metadata = mongoObject._metadata || {};\n    mongoObject._metadata.indexes = indexes;\n  }\n\n  if (!mongoObject._metadata) { // cleanup the unused _metadata\n    delete mongoObject._metadata;\n  }\n\n  return mongoObject;\n}\n\n\nexport class MongoStorageAdapter implements StorageAdapter {\n  // Private\n  _uri: string;\n  _collectionPrefix: string;\n  _mongoOptions: Object;\n  // Public\n  connectionPromise: Promise<any>;\n  database: any;\n  client: MongoClient;\n  _maxTimeMS: ?number;\n  canSortOnJoinTables: boolean;\n\n  constructor({\n    uri = defaults.DefaultMongoURI,\n    collectionPrefix = '',\n    mongoOptions = {},\n  }: any) {\n    this._uri = uri;\n    this._collectionPrefix = collectionPrefix;\n    this._mongoOptions = mongoOptions;\n\n    // MaxTimeMS is not a global MongoDB client option, it is applied per operation.\n    this._maxTimeMS = mongoOptions.maxTimeMS;\n    this.canSortOnJoinTables = true;\n    delete mongoOptions.maxTimeMS;\n  }\n\n  connect() {\n    if (this.connectionPromise) {\n      return this.connectionPromise;\n    }\n\n    // parsing and re-formatting causes the auth value (if there) to get URI\n    // encoded\n    const encodedUri = formatUrl(parseUrl(this._uri));\n\n    this.connectionPromise = MongoClient.connect(encodedUri, this._mongoOptions).then(client => {\n      // Starting mongoDB 3.0, the MongoClient.connect don't return a DB anymore but a client\n      // Fortunately, we can get back the options and use them to select the proper DB.\n      // https://github.com/mongodb/node-mongodb-native/blob/2c35d76f08574225b8db02d7bef687123e6bb018/lib/mongo_client.js#L885\n      const options = client.s.options;\n      const database = client.db(options.dbName);\n      if (!database) {\n        delete this.connectionPromise;\n        return;\n      }\n      database.on('error', () => {\n        delete this.connectionPromise;\n      });\n      database.on('close', () => {\n        delete this.connectionPromise;\n      });\n      this.client = client;\n      this.database = database;\n    }).catch((err) => {\n      delete this.connectionPromise;\n      return Promise.reject(err);\n    });\n\n    return this.connectionPromise;\n  }\n\n  handleError<T>(error: ?(Error | Parse.Error)): Promise<T> {\n    if (error && error.code === 13) { // Unauthorized error\n      delete this.client;\n      delete this.database;\n      delete this.connectionPromise;\n      logger.error('Received unauthorized error', { error: error });\n    }\n    throw error;\n  }\n\n  handleShutdown() {\n    if (!this.client) {\n      return;\n    }\n    this.client.close(false);\n  }\n\n  _adaptiveCollection(name: string) {\n    return this.connect()\n      .then(() => this.database.collection(this._collectionPrefix + name))\n      .then(rawCollection => new MongoCollection(rawCollection))\n      .catch(err => this.handleError(err));\n  }\n\n  _schemaCollection(): Promise<MongoSchemaCollection> {\n    return this.connect()\n      .then(() => this._adaptiveCollection(MongoSchemaCollectionName))\n      .then(collection => new MongoSchemaCollection(collection));\n  }\n\n  classExists(name: string) {\n    return this.connect().then(() => {\n      return this.database.listCollections({ name: this._collectionPrefix + name }).toArray();\n    }).then(collections => {\n      return collections.length > 0;\n    }).catch(err => this.handleError(err));\n  }\n\n  setClassLevelPermissions(className: string, CLPs: any): Promise<void> {\n    return this._schemaCollection()\n      .then(schemaCollection => schemaCollection.updateSchema(className, {\n        $set: { '_metadata.class_permissions': CLPs }\n      })).catch(err => this.handleError(err));\n  }\n\n  setIndexesWithSchemaFormat(className: string, submittedIndexes: any, existingIndexes: any = {}, fields: any): Promise<void> {\n    if (submittedIndexes === undefined) {\n      return Promise.resolve();\n    }\n    if (Object.keys(existingIndexes).length === 0) {\n      existingIndexes = { _id_: { _id: 1} };\n    }\n    const deletePromises = [];\n    const insertedIndexes = [];\n    Object.keys(submittedIndexes).forEach(name => {\n      const field = submittedIndexes[name];\n      if (existingIndexes[name] && field.__op !== 'Delete') {\n        throw new Parse.Error(Parse.Error.INVALID_QUERY, `Index ${name} exists, cannot update.`);\n      }\n      if (!existingIndexes[name] && field.__op === 'Delete') {\n        throw new Parse.Error(Parse.Error.INVALID_QUERY, `Index ${name} does not exist, cannot delete.`);\n      }\n      if (field.__op === 'Delete') {\n        const promise = this.dropIndex(className, name);\n        deletePromises.push(promise);\n        delete existingIndexes[name];\n      } else {\n        Object.keys(field).forEach(key => {\n          if (!fields.hasOwnProperty(key)) {\n            throw new Parse.Error(Parse.Error.INVALID_QUERY, `Field ${key} does not exist, cannot add index.`);\n          }\n        });\n        existingIndexes[name] = field;\n        insertedIndexes.push({\n          key: field,\n          name,\n        });\n      }\n    });\n    let insertPromise = Promise.resolve();\n    if (insertedIndexes.length > 0) {\n      insertPromise = this.createIndexes(className, insertedIndexes);\n    }\n    return Promise.all(deletePromises)\n      .then(() => insertPromise)\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.updateSchema(className, {\n        $set: { '_metadata.indexes':  existingIndexes }\n      }))\n      .catch(err => this.handleError(err));\n  }\n\n  setIndexesFromMongo(className: string) {\n    return this.getIndexes(className).then((indexes) => {\n      indexes = indexes.reduce((obj, index) => {\n        if (index.key._fts) {\n          delete index.key._fts;\n          delete index.key._ftsx;\n          for (const field in index.weights) {\n            index.key[field] = 'text';\n          }\n        }\n        obj[index.name] = index.key;\n        return obj;\n      }, {});\n      return this._schemaCollection()\n        .then(schemaCollection => schemaCollection.updateSchema(className, {\n          $set: { '_metadata.indexes': indexes }\n        }));\n    })\n      .catch(err => this.handleError(err))\n      .catch(() => {\n        // Ignore if collection not found\n        return Promise.resolve();\n      });\n  }\n\n  createClass(className: string, schema: SchemaType): Promise<void> {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoObject = mongoSchemaFromFieldsAndClassNameAndCLP(schema.fields, className, schema.classLevelPermissions, schema.indexes);\n    mongoObject._id = className;\n    return this.setIndexesWithSchemaFormat(className, schema.indexes, {}, schema.fields)\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.insertSchema(mongoObject))\n      .catch(err => this.handleError(err));\n  }\n\n  addFieldIfNotExists(className: string, fieldName: string, type: any): Promise<void> {\n    return this._schemaCollection()\n      .then(schemaCollection => schemaCollection.addFieldIfNotExists(className, fieldName, type))\n      .then(() => this.createIndexesIfNeeded(className, fieldName, type))\n      .catch(err => this.handleError(err));\n  }\n\n  // Drops a collection. Resolves with true if it was a Parse Schema (eg. _User, Custom, etc.)\n  // and resolves with false if it wasn't (eg. a join table). Rejects if deletion was impossible.\n  deleteClass(className: string) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection.drop())\n      .catch(error => {\n      // 'ns not found' means collection was already gone. Ignore deletion attempt.\n        if (error.message == 'ns not found') {\n          return;\n        }\n        throw error;\n      })\n    // We've dropped the collection, now remove the _SCHEMA document\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.findAndDeleteSchema(className))\n      .catch(err => this.handleError(err));\n  }\n\n  deleteAllClasses(fast: boolean) {\n    return storageAdapterAllCollections(this)\n      .then(collections => Promise.all(collections.map(collection => fast ? collection.remove({}) : collection.drop())));\n  }\n\n  // Remove the column and all the data. For Relations, the _Join collection is handled\n  // specially, this function does not delete _Join columns. It should, however, indicate\n  // that the relation fields does not exist anymore. In mongo, this means removing it from\n  // the _SCHEMA collection.  There should be no actual data in the collection under the same name\n  // as the relation column, so it's fine to attempt to delete it. If the fields listed to be\n  // deleted do not exist, this function should return successfully anyways. Checking for\n  // attempts to delete non-existent fields is the responsibility of Parse Server.\n\n  // Pointer field names are passed for legacy reasons: the original mongo\n  // format stored pointer field names differently in the database, and therefore\n  // needed to know the type of the field before it could delete it. Future database\n  // adapters should ignore the pointerFieldNames argument. All the field names are in\n  // fieldNames, they show up additionally in the pointerFieldNames database for use\n  // by the mongo adapter, which deals with the legacy mongo format.\n\n  // This function is not obligated to delete fields atomically. It is given the field\n  // names in a list so that databases that are capable of deleting fields atomically\n  // may do so.\n\n  // Returns a Promise.\n  deleteFields(className: string, schema: SchemaType, fieldNames: string[]) {\n    const mongoFormatNames = fieldNames.map(fieldName => {\n      if (schema.fields[fieldName].type === 'Pointer') {\n        return `_p_${fieldName}`\n      } else {\n        return fieldName;\n      }\n    });\n    const collectionUpdate = { '$unset' : {} };\n    mongoFormatNames.forEach(name => {\n      collectionUpdate['$unset'][name] = null;\n    });\n\n    const schemaUpdate = { '$unset' : {} };\n    fieldNames.forEach(name => {\n      schemaUpdate['$unset'][name] = null;\n    });\n\n    return this._adaptiveCollection(className)\n      .then(collection => collection.updateMany({}, collectionUpdate))\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.updateSchema(className, schemaUpdate))\n      .catch(err => this.handleError(err));\n  }\n\n  // Return a promise for all schemas known to this adapter, in Parse format. In case the\n  // schemas cannot be retrieved, returns a promise that rejects. Requirements for the\n  // rejection reason are TBD.\n  getAllClasses(): Promise<StorageClass[]> {\n    return this._schemaCollection().then(schemasCollection => schemasCollection._fetchAllSchemasFrom_SCHEMA())\n      .catch(err => this.handleError(err));\n  }\n\n  // Return a promise for the schema with the given name, in Parse format. If\n  // this adapter doesn't know about the schema, return a promise that rejects with\n  // undefined as the reason.\n  getClass(className: string): Promise<StorageClass> {\n    return this._schemaCollection()\n      .then(schemasCollection => schemasCollection._fetchOneSchemaFrom_SCHEMA(className))\n      .catch(err => this.handleError(err));\n  }\n\n  // TODO: As yet not particularly well specified. Creates an object. Maybe shouldn't even need the schema,\n  // and should infer from the type. Or maybe does need the schema for validations. Or maybe needs\n  // the schema only for the legacy mongo format. We'll figure that out later.\n  createObject(className: string, schema: SchemaType, object: any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoObject = parseObjectToMongoObjectForCreate(className, object, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.insertOne(mongoObject))\n      .catch(error => {\n        if (error.code === 11000) { // Duplicate value\n          const err = new Parse.Error(Parse.Error.DUPLICATE_VALUE, 'A duplicate value for a field with unique values was provided');\n          err.underlyingError = error;\n          if (error.message) {\n            const matches = error.message.match(/index:[\\sa-zA-Z0-9_\\-\\.]+\\$?([a-zA-Z_-]+)_1/);\n            if (matches && Array.isArray(matches)) {\n              err.userInfo = { duplicated_field: matches[1] };\n            }\n          }\n          throw err;\n        }\n        throw error;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  // Remove all objects that match the given Parse Query.\n  // If no objects match, reject with OBJECT_NOT_FOUND. If objects are found and deleted, resolve with undefined.\n  // If there is some other error, reject with INTERNAL_SERVER_ERROR.\n  deleteObjectsByQuery(className: string, schema: SchemaType, query: QueryType) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    return this._adaptiveCollection(className)\n      .then(collection => {\n        const mongoWhere = transformWhere(className, query, schema);\n        return collection.deleteMany(mongoWhere)\n      })\n      .catch(err => this.handleError(err))\n      .then(({ result }) => {\n        if (result.n === 0) {\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n        }\n        return Promise.resolve();\n      }, () => {\n        throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'Database adapter error');\n      });\n  }\n\n  // Apply the update to all objects that match the given Parse Query.\n  updateObjectsByQuery(className: string, schema: SchemaType, query: QueryType, update: any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoUpdate = transformUpdate(className, update, schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.updateMany(mongoWhere, mongoUpdate))\n      .catch(err => this.handleError(err));\n  }\n\n  // Atomically finds and updates an object based on query.\n  // Return value not currently well specified.\n  findOneAndUpdate(className: string, schema: SchemaType, query: QueryType, update: any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoUpdate = transformUpdate(className, update, schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.findAndModify(mongoWhere, [], mongoUpdate, { new: true }))\n      .then(result => mongoObjectToParseObject(className, result.value, schema))\n      .catch(error => {\n        if (error.code === 11000) {\n          throw new Parse.Error(Parse.Error.DUPLICATE_VALUE, 'A duplicate value for a field with unique values was provided');\n        }\n        throw error;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  // Hopefully we can get rid of this. It's only used for config and hooks.\n  upsertOneObject(className: string, schema: SchemaType, query: QueryType, update: any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoUpdate = transformUpdate(className, update, schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.upsertOne(mongoWhere, mongoUpdate))\n      .catch(err => this.handleError(err));\n  }\n\n  // Executes a find. Accepts: className, query in Parse format, and { skip, limit, sort }.\n  find(className: string, schema: SchemaType, query: QueryType, { skip, limit, sort, keys, readPreference }: QueryOptions): Promise<any> {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    const mongoSort = _.mapKeys(sort, (value, fieldName) => transformKey(className, fieldName, schema));\n    const mongoKeys = _.reduce(keys, (memo, key) => {\n      memo[transformKey(className, key, schema)] = 1;\n      return memo;\n    }, {});\n\n    readPreference = this._parseReadPreference(readPreference);\n    return this.createTextIndexesIfNeeded(className, query, schema)\n      .then(() => this._adaptiveCollection(className))\n      .then(collection => collection.find(mongoWhere, {\n        skip,\n        limit,\n        sort: mongoSort,\n        keys: mongoKeys,\n        maxTimeMS: this._maxTimeMS,\n        readPreference,\n      }))\n      .then(objects => objects.map(object => mongoObjectToParseObject(className, object, schema)))\n      .catch(err => this.handleError(err));\n  }\n\n  // Create a unique index. Unique indexes on nullable fields are not allowed. Since we don't\n  // currently know which fields are nullable and which aren't, we ignore that criteria.\n  // As such, we shouldn't expose this function to users of parse until we have an out-of-band\n  // Way of determining if a field is nullable. Undefined doesn't count against uniqueness,\n  // which is why we use sparse indexes.\n  ensureUniqueness(className: string, schema: SchemaType, fieldNames: string[]) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const indexCreationRequest = {};\n    const mongoFieldNames = fieldNames.map(fieldName => transformKey(className, fieldName, schema));\n    mongoFieldNames.forEach(fieldName => {\n      indexCreationRequest[fieldName] = 1;\n    });\n    return this._adaptiveCollection(className)\n      .then(collection => collection._ensureSparseUniqueIndexInBackground(indexCreationRequest))\n      .catch(error => {\n        if (error.code === 11000) {\n          throw new Parse.Error(Parse.Error.DUPLICATE_VALUE, 'Tried to ensure field uniqueness for a class that already has duplicates.');\n        }\n        throw error;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  // Used in tests\n  _rawFind(className: string, query: QueryType) {\n    return this._adaptiveCollection(className).then(collection => collection.find(query, {\n      maxTimeMS: this._maxTimeMS,\n    })).catch(err => this.handleError(err));\n  }\n\n  // Executes a count.\n  count(className: string, schema: SchemaType, query: QueryType, readPreference: ?string) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    readPreference = this._parseReadPreference(readPreference);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.count(transformWhere(className, query, schema), {\n        maxTimeMS: this._maxTimeMS,\n        readPreference,\n      }))\n      .catch(err => this.handleError(err));\n  }\n\n  distinct(className: string, schema: SchemaType, query: QueryType, fieldName: string) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const isPointerField = schema.fields[fieldName] && schema.fields[fieldName].type === 'Pointer';\n    if (isPointerField) {\n      fieldName = `_p_${fieldName}`\n    }\n    return this._adaptiveCollection(className)\n      .then(collection => collection.distinct(fieldName, transformWhere(className, query, schema)))\n      .then(objects => {\n        objects = objects.filter((obj) => obj != null);\n        return objects.map(object => {\n          if (isPointerField) {\n            const field = fieldName.substring(3);\n            return transformPointerString(schema, field, object);\n          }\n          return mongoObjectToParseObject(className, object, schema);\n        });\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  aggregate(className: string, schema: any, pipeline: any, readPreference: ?string) {\n    let isPointerField = false;\n    pipeline = pipeline.map((stage) => {\n      if (stage.$group) {\n        stage.$group = this._parseAggregateGroupArgs(schema, stage.$group);\n        if (stage.$group._id && (typeof stage.$group._id === 'string') && stage.$group._id.indexOf('$_p_') >= 0) {\n          isPointerField = true;\n        }\n      }\n      if (stage.$match) {\n        stage.$match = this._parseAggregateArgs(schema, stage.$match);\n      }\n      if (stage.$project) {\n        stage.$project = this._parseAggregateProjectArgs(schema, stage.$project);\n      }\n      return stage;\n    });\n    readPreference = this._parseReadPreference(readPreference);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.aggregate(pipeline, { readPreference, maxTimeMS: this._maxTimeMS }))\n      .catch(error => {\n        if (error.code === 16006) {\n          throw new Parse.Error(Parse.Error.INVALID_QUERY, error.message);\n        }\n        throw error;\n      })\n      .then(results => {\n        results.forEach(result => {\n          if (result.hasOwnProperty('_id')) {\n            if (isPointerField && result._id) {\n              result._id = result._id.split('$')[1];\n            }\n            if (result._id == null || _.isEmpty(result._id)) {\n              result._id = null;\n            }\n            result.objectId = result._id;\n            delete result._id;\n          }\n        });\n        return results;\n      })\n      .then(objects => objects.map(object => mongoObjectToParseObject(className, object, schema)))\n      .catch(err => this.handleError(err));\n  }\n\n  // This function will recursively traverse the pipeline and convert any Pointer or Date columns.\n  // If we detect a pointer column we will rename the column being queried for to match the column\n  // in the database. We also modify the value to what we expect the value to be in the database\n  // as well.\n  // For dates, the driver expects a Date object, but we have a string coming in. So we'll convert\n  // the string to a Date so the driver can perform the necessary comparison.\n  //\n  // The goal of this method is to look for the \"leaves\" of the pipeline and determine if it needs\n  // to be converted. The pipeline can have a few different forms. For more details, see:\n  //     https://docs.mongodb.com/manual/reference/operator/aggregation/\n  //\n  // If the pipeline is an array, it means we are probably parsing an '$and' or '$or' operator. In\n  // that case we need to loop through all of it's children to find the columns being operated on.\n  // If the pipeline is an object, then we'll loop through the keys checking to see if the key name\n  // matches one of the schema columns. If it does match a column and the column is a Pointer or\n  // a Date, then we'll convert the value as described above.\n  //\n  // As much as I hate recursion...this seemed like a good fit for it. We're essentially traversing\n  // down a tree to find a \"leaf node\" and checking to see if it needs to be converted.\n  _parseAggregateArgs(schema: any, pipeline: any): any {\n    if (Array.isArray(pipeline)) {\n      return pipeline.map((value) => this._parseAggregateArgs(schema, value));\n    } else if (typeof pipeline === 'object') {\n      const returnValue = {};\n      for (const field in pipeline) {\n        if (schema.fields[field] && schema.fields[field].type === 'Pointer') {\n          if (typeof pipeline[field] === 'object') {\n            // Pass objects down to MongoDB...this is more than likely an $exists operator.\n            returnValue[`_p_${field}`] = pipeline[field];\n          } else {\n            returnValue[`_p_${field}`] = `${schema.fields[field].targetClass}$${pipeline[field]}`;\n          }\n        } else if (schema.fields[field] && schema.fields[field].type === 'Date') {\n          returnValue[field] = this._convertToDate(pipeline[field]);\n        } else {\n          returnValue[field] = this._parseAggregateArgs(schema, pipeline[field]);\n        }\n\n        if (field === 'objectId') {\n          returnValue['_id'] = returnValue[field];\n          delete returnValue[field];\n        } else if (field === 'createdAt') {\n          returnValue['_created_at'] = returnValue[field];\n          delete returnValue[field];\n        } else if (field === 'updatedAt') {\n          returnValue['_updated_at'] = returnValue[field];\n          delete returnValue[field];\n        }\n      }\n      return returnValue;\n    }\n    return pipeline;\n  }\n\n  // This function is slightly different than the one above. Rather than trying to combine these\n  // two functions and making the code even harder to understand, I decided to split it up. The\n  // difference with this function is we are not transforming the values, only the keys of the\n  // pipeline.\n  _parseAggregateProjectArgs(schema: any, pipeline: any): any {\n    const returnValue = {};\n    for (const field in pipeline) {\n      if (schema.fields[field] && schema.fields[field].type === 'Pointer') {\n        returnValue[`_p_${field}`] = pipeline[field];\n      } else {\n        returnValue[field] = this._parseAggregateArgs(schema, pipeline[field]);\n      }\n\n      if (field === 'objectId') {\n        returnValue['_id'] = returnValue[field];\n        delete returnValue[field];\n      } else if (field === 'createdAt') {\n        returnValue['_created_at'] = returnValue[field];\n        delete returnValue[field];\n      } else if (field === 'updatedAt') {\n        returnValue['_updated_at'] = returnValue[field];\n        delete returnValue[field];\n      }\n    }\n    return returnValue;\n  }\n\n  // This function is slightly different than the two above. MongoDB $group aggregate looks like:\n  //     { $group: { _id: <expression>, <field1>: { <accumulator1> : <expression1> }, ... } }\n  // The <expression> could be a column name, prefixed with the '$' character. We'll look for\n  // these <expression> and check to see if it is a 'Pointer' or if it's one of createdAt,\n  // updatedAt or objectId and change it accordingly.\n  _parseAggregateGroupArgs(schema: any, pipeline: any): any {\n    if (Array.isArray(pipeline)) {\n      return pipeline.map((value) => this._parseAggregateGroupArgs(schema, value));\n    } else if (typeof pipeline === 'object') {\n      const returnValue = {};\n      for (const field in pipeline) {\n        returnValue[field] = this._parseAggregateGroupArgs(schema, pipeline[field]);\n      }\n      return returnValue;\n    } else if (typeof pipeline === 'string') {\n      const field = pipeline.substring(1);\n      if (schema.fields[field] && schema.fields[field].type === 'Pointer') {\n        return `$_p_${field}`;\n      } else if (field == 'createdAt') {\n        return '$_created_at';\n      } else if (field == 'updatedAt') {\n        return '$_updated_at';\n      }\n    }\n    return pipeline;\n  }\n\n  // This function will attempt to convert the provided value to a Date object. Since this is part\n  // of an aggregation pipeline, the value can either be a string or it can be another object with\n  // an operator in it (like $gt, $lt, etc). Because of this I felt it was easier to make this a\n  // recursive method to traverse down to the \"leaf node\" which is going to be the string.\n  _convertToDate(value: any): any {\n    if (typeof value === 'string') {\n      return new Date(value);\n    }\n\n    const returnValue = {}\n    for (const field in value) {\n      returnValue[field] = this._convertToDate(value[field])\n    }\n    return returnValue;\n  }\n\n  _parseReadPreference(readPreference: ?string): ?string {\n    switch (readPreference) {\n    case 'PRIMARY':\n      readPreference = ReadPreference.PRIMARY;\n      break;\n    case 'PRIMARY_PREFERRED':\n      readPreference = ReadPreference.PRIMARY_PREFERRED;\n      break;\n    case 'SECONDARY':\n      readPreference = ReadPreference.SECONDARY;\n      break;\n    case 'SECONDARY_PREFERRED':\n      readPreference = ReadPreference.SECONDARY_PREFERRED;\n      break;\n    case 'NEAREST':\n      readPreference = ReadPreference.NEAREST;\n      break;\n    case undefined:\n      break;\n    default:\n      throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Not supported read preference.');\n    }\n    return readPreference;\n  }\n\n  performInitialization(): Promise<void> {\n    return Promise.resolve();\n  }\n\n  createIndex(className: string, index: any) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.createIndex(index, {background: true}))\n      .catch(err => this.handleError(err));\n  }\n\n  createIndexes(className: string, indexes: any) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.createIndexes(indexes, {background: true}))\n      .catch(err => this.handleError(err));\n  }\n\n  createIndexesIfNeeded(className: string, fieldName: string, type: any) {\n    if (type && type.type === 'Polygon') {\n      const index = {\n        [fieldName]: '2dsphere'\n      };\n      return this.createIndex(className, index);\n    }\n    return Promise.resolve();\n  }\n\n  createTextIndexesIfNeeded(className: string, query: QueryType, schema: any): Promise<void> {\n    for(const fieldName in query) {\n      if (!query[fieldName] || !query[fieldName].$text) {\n        continue;\n      }\n      const existingIndexes = schema.indexes;\n      for (const key in existingIndexes) {\n        const index = existingIndexes[key];\n        if (index.hasOwnProperty(fieldName)) {\n          return Promise.resolve();\n        }\n      }\n      const indexName = `${fieldName}_text`;\n      const textIndex = {\n        [indexName]: { [fieldName]: 'text' }\n      };\n      return this.setIndexesWithSchemaFormat(className, textIndex, existingIndexes, schema.fields)\n        .catch((error) => {\n          if (error.code === 85) { // Index exist with different options\n            return this.setIndexesFromMongo(className);\n          }\n          throw error;\n        });\n    }\n    return Promise.resolve();\n  }\n\n  getIndexes(className: string) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.indexes())\n      .catch(err => this.handleError(err));\n  }\n\n  dropIndex(className: string, index: any) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.dropIndex(index))\n      .catch(err => this.handleError(err));\n  }\n\n  dropAllIndexes(className: string) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.dropIndexes())\n      .catch(err => this.handleError(err));\n  }\n\n  updateSchemaWithIndexes(): Promise<any> {\n    return this.getAllClasses()\n      .then((classes) => {\n        const promises = classes.map((schema) => {\n          return this.setIndexesFromMongo(schema.className);\n        });\n        return Promise.all(promises);\n      })\n      .catch(err => this.handleError(err));\n  }\n}\n\nexport default MongoStorageAdapter;\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/Adapters/Storage/Mongo/MongoStorageAdapter.js"],"names":["mongodb","require","MongoClient","ReadPreference","MongoSchemaCollectionName","storageAdapterAllCollections","mongoAdapter","connect","then","database","collections","filter","collection","namespace","match","collectionName","indexOf","_collectionPrefix","convertParseSchemaToMongoSchema","schema","fields","_rperm","_wperm","className","_hashed_password","mongoSchemaFromFieldsAndClassNameAndCLP","classLevelPermissions","indexes","mongoObject","_id","objectId","updatedAt","createdAt","_metadata","undefined","fieldName","MongoSchemaCollection","parseFieldTypeToMongoFieldType","class_permissions","Object","keys","length","MongoStorageAdapter","constructor","uri","defaults","DefaultMongoURI","collectionPrefix","mongoOptions","_uri","_mongoOptions","useNewUrlParser","_maxTimeMS","maxTimeMS","canSortOnJoinTables","connectionPromise","encodedUri","client","options","s","db","dbName","on","catch","err","Promise","reject","handleError","error","code","logger","handleShutdown","close","_adaptiveCollection","name","rawCollection","MongoCollection","_schemaCollection","classExists","listCollections","toArray","setClassLevelPermissions","CLPs","schemaCollection","updateSchema","$set","setIndexesWithSchemaFormat","submittedIndexes","existingIndexes","resolve","_id_","deletePromises","insertedIndexes","forEach","field","__op","Parse","Error","INVALID_QUERY","promise","dropIndex","push","key","hasOwnProperty","insertPromise","createIndexes","all","setIndexesFromMongo","getIndexes","reduce","obj","index","_fts","_ftsx","weights","createClass","insertSchema","addFieldIfNotExists","type","createIndexesIfNeeded","deleteClass","drop","message","findAndDeleteSchema","deleteAllClasses","fast","map","remove","deleteFields","fieldNames","mongoFormatNames","collectionUpdate","schemaUpdate","updateMany","getAllClasses","schemasCollection","_fetchAllSchemasFrom_SCHEMA","getClass","_fetchOneSchemaFrom_SCHEMA","createObject","object","insertOne","DUPLICATE_VALUE","underlyingError","matches","Array","isArray","userInfo","duplicated_field","deleteObjectsByQuery","query","mongoWhere","deleteMany","result","n","OBJECT_NOT_FOUND","INTERNAL_SERVER_ERROR","updateObjectsByQuery","update","mongoUpdate","findOneAndUpdate","_mongoCollection","findAndModify","new","value","upsertOneObject","upsertOne","find","skip","limit","sort","readPreference","mongoSort","_","mapKeys","mongoKeys","memo","_parseReadPreference","createTextIndexesIfNeeded","objects","ensureUniqueness","indexCreationRequest","mongoFieldNames","_ensureSparseUniqueIndexInBackground","_rawFind","count","distinct","isPointerField","substring","aggregate","pipeline","stage","$group","_parseAggregateGroupArgs","$match","_parseAggregateArgs","$project","_parseAggregateProjectArgs","results","split","isEmpty","returnValue","targetClass","_convertToDate","Date","PRIMARY","PRIMARY_PREFERRED","SECONDARY","SECONDARY_PREFERRED","NEAREST","performInitialization","createIndex","background","$text","indexName","textIndex","dropAllIndexes","dropIndexes","updateSchemaWithIndexes","classes","promises"],"mappings":"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","file":"MongoStorageAdapter.js","sourcesContent":["// @flow\nimport MongoCollection       from './MongoCollection';\nimport MongoSchemaCollection from './MongoSchemaCollection';\nimport { StorageAdapter }    from '../StorageAdapter';\nimport type { SchemaType,\n  QueryType,\n  StorageClass,\n  QueryOptions } from '../StorageAdapter';\nimport {\n  parse as parseUrl,\n  format as formatUrl,\n} from '../../../vendor/mongodbUrl';\nimport {\n  parseObjectToMongoObjectForCreate,\n  mongoObjectToParseObject,\n  transformKey,\n  transformWhere,\n  transformUpdate,\n  transformPointerString,\n} from './MongoTransform';\n// @flow-disable-next\nimport Parse                 from 'parse/node';\n// @flow-disable-next\nimport _                     from 'lodash';\nimport defaults              from '../../../defaults';\nimport logger                from '../../../logger';\n\n// @flow-disable-next\nconst mongodb = require('mongodb');\nconst MongoClient = mongodb.MongoClient;\nconst ReadPreference = mongodb.ReadPreference;\n\nconst MongoSchemaCollectionName = '_SCHEMA';\n\nconst storageAdapterAllCollections = mongoAdapter => {\n  return mongoAdapter.connect()\n    .then(() => mongoAdapter.database.collections())\n    .then(collections => {\n      return collections.filter(collection => {\n        if (collection.namespace.match(/\\.system\\./)) {\n          return false;\n        }\n        // TODO: If you have one app with a collection prefix that happens to be a prefix of another\n        // apps prefix, this will go very very badly. We should fix that somehow.\n        return (collection.collectionName.indexOf(mongoAdapter._collectionPrefix) == 0);\n      });\n    });\n}\n\nconst convertParseSchemaToMongoSchema = ({...schema}) => {\n  delete schema.fields._rperm;\n  delete schema.fields._wperm;\n\n  if (schema.className === '_User') {\n    // Legacy mongo adapter knows about the difference between password and _hashed_password.\n    // Future database adapters will only know about _hashed_password.\n    // Note: Parse Server will bring back password with injectDefaultSchema, so we don't need\n    // to add _hashed_password back ever.\n    delete schema.fields._hashed_password;\n  }\n\n  return schema;\n}\n\n// Returns { code, error } if invalid, or { result }, an object\n// suitable for inserting into _SCHEMA collection, otherwise.\nconst mongoSchemaFromFieldsAndClassNameAndCLP = (fields, className, classLevelPermissions, indexes) => {\n  const mongoObject = {\n    _id: className,\n    objectId: 'string',\n    updatedAt: 'string',\n    createdAt: 'string',\n    _metadata: undefined,\n  };\n\n  for (const fieldName in fields) {\n    mongoObject[fieldName] = MongoSchemaCollection.parseFieldTypeToMongoFieldType(fields[fieldName]);\n  }\n\n  if (typeof classLevelPermissions !== 'undefined') {\n    mongoObject._metadata = mongoObject._metadata || {};\n    if (!classLevelPermissions) {\n      delete mongoObject._metadata.class_permissions;\n    } else {\n      mongoObject._metadata.class_permissions = classLevelPermissions;\n    }\n  }\n\n  if (indexes && typeof indexes === 'object' && Object.keys(indexes).length > 0) {\n    mongoObject._metadata = mongoObject._metadata || {};\n    mongoObject._metadata.indexes = indexes;\n  }\n\n  if (!mongoObject._metadata) { // cleanup the unused _metadata\n    delete mongoObject._metadata;\n  }\n\n  return mongoObject;\n}\n\n\nexport class MongoStorageAdapter implements StorageAdapter {\n  // Private\n  _uri: string;\n  _collectionPrefix: string;\n  _mongoOptions: Object;\n  // Public\n  connectionPromise: Promise<any>;\n  database: any;\n  client: MongoClient;\n  _maxTimeMS: ?number;\n  canSortOnJoinTables: boolean;\n\n  constructor({\n    uri = defaults.DefaultMongoURI,\n    collectionPrefix = '',\n    mongoOptions = {},\n  }: any) {\n    this._uri = uri;\n    this._collectionPrefix = collectionPrefix;\n    this._mongoOptions = mongoOptions;\n    this._mongoOptions.useNewUrlParser = true;\n\n    // MaxTimeMS is not a global MongoDB client option, it is applied per operation.\n    this._maxTimeMS = mongoOptions.maxTimeMS;\n    this.canSortOnJoinTables = true;\n    delete mongoOptions.maxTimeMS;\n  }\n\n  connect() {\n    if (this.connectionPromise) {\n      return this.connectionPromise;\n    }\n\n    // parsing and re-formatting causes the auth value (if there) to get URI\n    // encoded\n    const encodedUri = formatUrl(parseUrl(this._uri));\n\n    this.connectionPromise = MongoClient.connect(encodedUri, this._mongoOptions).then(client => {\n      // Starting mongoDB 3.0, the MongoClient.connect don't return a DB anymore but a client\n      // Fortunately, we can get back the options and use them to select the proper DB.\n      // https://github.com/mongodb/node-mongodb-native/blob/2c35d76f08574225b8db02d7bef687123e6bb018/lib/mongo_client.js#L885\n      const options = client.s.options;\n      const database = client.db(options.dbName);\n      if (!database) {\n        delete this.connectionPromise;\n        return;\n      }\n      database.on('error', () => {\n        delete this.connectionPromise;\n      });\n      database.on('close', () => {\n        delete this.connectionPromise;\n      });\n      this.client = client;\n      this.database = database;\n    }).catch((err) => {\n      delete this.connectionPromise;\n      return Promise.reject(err);\n    });\n\n    return this.connectionPromise;\n  }\n\n  handleError<T>(error: ?(Error | Parse.Error)): Promise<T> {\n    if (error && error.code === 13) { // Unauthorized error\n      delete this.client;\n      delete this.database;\n      delete this.connectionPromise;\n      logger.error('Received unauthorized error', { error: error });\n    }\n    throw error;\n  }\n\n  handleShutdown() {\n    if (!this.client) {\n      return;\n    }\n    this.client.close(false);\n  }\n\n  _adaptiveCollection(name: string) {\n    return this.connect()\n      .then(() => this.database.collection(this._collectionPrefix + name))\n      .then(rawCollection => new MongoCollection(rawCollection))\n      .catch(err => this.handleError(err));\n  }\n\n  _schemaCollection(): Promise<MongoSchemaCollection> {\n    return this.connect()\n      .then(() => this._adaptiveCollection(MongoSchemaCollectionName))\n      .then(collection => new MongoSchemaCollection(collection));\n  }\n\n  classExists(name: string) {\n    return this.connect().then(() => {\n      return this.database.listCollections({ name: this._collectionPrefix + name }).toArray();\n    }).then(collections => {\n      return collections.length > 0;\n    }).catch(err => this.handleError(err));\n  }\n\n  setClassLevelPermissions(className: string, CLPs: any): Promise<void> {\n    return this._schemaCollection()\n      .then(schemaCollection => schemaCollection.updateSchema(className, {\n        $set: { '_metadata.class_permissions': CLPs }\n      })).catch(err => this.handleError(err));\n  }\n\n  setIndexesWithSchemaFormat(className: string, submittedIndexes: any, existingIndexes: any = {}, fields: any): Promise<void> {\n    if (submittedIndexes === undefined) {\n      return Promise.resolve();\n    }\n    if (Object.keys(existingIndexes).length === 0) {\n      existingIndexes = { _id_: { _id: 1} };\n    }\n    const deletePromises = [];\n    const insertedIndexes = [];\n    Object.keys(submittedIndexes).forEach(name => {\n      const field = submittedIndexes[name];\n      if (existingIndexes[name] && field.__op !== 'Delete') {\n        throw new Parse.Error(Parse.Error.INVALID_QUERY, `Index ${name} exists, cannot update.`);\n      }\n      if (!existingIndexes[name] && field.__op === 'Delete') {\n        throw new Parse.Error(Parse.Error.INVALID_QUERY, `Index ${name} does not exist, cannot delete.`);\n      }\n      if (field.__op === 'Delete') {\n        const promise = this.dropIndex(className, name);\n        deletePromises.push(promise);\n        delete existingIndexes[name];\n      } else {\n        Object.keys(field).forEach(key => {\n          if (!fields.hasOwnProperty(key)) {\n            throw new Parse.Error(Parse.Error.INVALID_QUERY, `Field ${key} does not exist, cannot add index.`);\n          }\n        });\n        existingIndexes[name] = field;\n        insertedIndexes.push({\n          key: field,\n          name,\n        });\n      }\n    });\n    let insertPromise = Promise.resolve();\n    if (insertedIndexes.length > 0) {\n      insertPromise = this.createIndexes(className, insertedIndexes);\n    }\n    return Promise.all(deletePromises)\n      .then(() => insertPromise)\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.updateSchema(className, {\n        $set: { '_metadata.indexes':  existingIndexes }\n      }))\n      .catch(err => this.handleError(err));\n  }\n\n  setIndexesFromMongo(className: string) {\n    return this.getIndexes(className).then((indexes) => {\n      indexes = indexes.reduce((obj, index) => {\n        if (index.key._fts) {\n          delete index.key._fts;\n          delete index.key._ftsx;\n          for (const field in index.weights) {\n            index.key[field] = 'text';\n          }\n        }\n        obj[index.name] = index.key;\n        return obj;\n      }, {});\n      return this._schemaCollection()\n        .then(schemaCollection => schemaCollection.updateSchema(className, {\n          $set: { '_metadata.indexes': indexes }\n        }));\n    })\n      .catch(err => this.handleError(err))\n      .catch(() => {\n        // Ignore if collection not found\n        return Promise.resolve();\n      });\n  }\n\n  createClass(className: string, schema: SchemaType): Promise<void> {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoObject = mongoSchemaFromFieldsAndClassNameAndCLP(schema.fields, className, schema.classLevelPermissions, schema.indexes);\n    mongoObject._id = className;\n    return this.setIndexesWithSchemaFormat(className, schema.indexes, {}, schema.fields)\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.insertSchema(mongoObject))\n      .catch(err => this.handleError(err));\n  }\n\n  addFieldIfNotExists(className: string, fieldName: string, type: any): Promise<void> {\n    return this._schemaCollection()\n      .then(schemaCollection => schemaCollection.addFieldIfNotExists(className, fieldName, type))\n      .then(() => this.createIndexesIfNeeded(className, fieldName, type))\n      .catch(err => this.handleError(err));\n  }\n\n  // Drops a collection. Resolves with true if it was a Parse Schema (eg. _User, Custom, etc.)\n  // and resolves with false if it wasn't (eg. a join table). Rejects if deletion was impossible.\n  deleteClass(className: string) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection.drop())\n      .catch(error => {\n      // 'ns not found' means collection was already gone. Ignore deletion attempt.\n        if (error.message == 'ns not found') {\n          return;\n        }\n        throw error;\n      })\n    // We've dropped the collection, now remove the _SCHEMA document\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.findAndDeleteSchema(className))\n      .catch(err => this.handleError(err));\n  }\n\n  deleteAllClasses(fast: boolean) {\n    return storageAdapterAllCollections(this)\n      .then(collections => Promise.all(collections.map(collection => fast ? collection.remove({}) : collection.drop())));\n  }\n\n  // Remove the column and all the data. For Relations, the _Join collection is handled\n  // specially, this function does not delete _Join columns. It should, however, indicate\n  // that the relation fields does not exist anymore. In mongo, this means removing it from\n  // the _SCHEMA collection.  There should be no actual data in the collection under the same name\n  // as the relation column, so it's fine to attempt to delete it. If the fields listed to be\n  // deleted do not exist, this function should return successfully anyways. Checking for\n  // attempts to delete non-existent fields is the responsibility of Parse Server.\n\n  // Pointer field names are passed for legacy reasons: the original mongo\n  // format stored pointer field names differently in the database, and therefore\n  // needed to know the type of the field before it could delete it. Future database\n  // adapters should ignore the pointerFieldNames argument. All the field names are in\n  // fieldNames, they show up additionally in the pointerFieldNames database for use\n  // by the mongo adapter, which deals with the legacy mongo format.\n\n  // This function is not obligated to delete fields atomically. It is given the field\n  // names in a list so that databases that are capable of deleting fields atomically\n  // may do so.\n\n  // Returns a Promise.\n  deleteFields(className: string, schema: SchemaType, fieldNames: string[]) {\n    const mongoFormatNames = fieldNames.map(fieldName => {\n      if (schema.fields[fieldName].type === 'Pointer') {\n        return `_p_${fieldName}`\n      } else {\n        return fieldName;\n      }\n    });\n    const collectionUpdate = { '$unset' : {} };\n    mongoFormatNames.forEach(name => {\n      collectionUpdate['$unset'][name] = null;\n    });\n\n    const schemaUpdate = { '$unset' : {} };\n    fieldNames.forEach(name => {\n      schemaUpdate['$unset'][name] = null;\n    });\n\n    return this._adaptiveCollection(className)\n      .then(collection => collection.updateMany({}, collectionUpdate))\n      .then(() => this._schemaCollection())\n      .then(schemaCollection => schemaCollection.updateSchema(className, schemaUpdate))\n      .catch(err => this.handleError(err));\n  }\n\n  // Return a promise for all schemas known to this adapter, in Parse format. In case the\n  // schemas cannot be retrieved, returns a promise that rejects. Requirements for the\n  // rejection reason are TBD.\n  getAllClasses(): Promise<StorageClass[]> {\n    return this._schemaCollection().then(schemasCollection => schemasCollection._fetchAllSchemasFrom_SCHEMA())\n      .catch(err => this.handleError(err));\n  }\n\n  // Return a promise for the schema with the given name, in Parse format. If\n  // this adapter doesn't know about the schema, return a promise that rejects with\n  // undefined as the reason.\n  getClass(className: string): Promise<StorageClass> {\n    return this._schemaCollection()\n      .then(schemasCollection => schemasCollection._fetchOneSchemaFrom_SCHEMA(className))\n      .catch(err => this.handleError(err));\n  }\n\n  // TODO: As yet not particularly well specified. Creates an object. Maybe shouldn't even need the schema,\n  // and should infer from the type. Or maybe does need the schema for validations. Or maybe needs\n  // the schema only for the legacy mongo format. We'll figure that out later.\n  createObject(className: string, schema: SchemaType, object: any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoObject = parseObjectToMongoObjectForCreate(className, object, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.insertOne(mongoObject))\n      .catch(error => {\n        if (error.code === 11000) { // Duplicate value\n          const err = new Parse.Error(Parse.Error.DUPLICATE_VALUE, 'A duplicate value for a field with unique values was provided');\n          err.underlyingError = error;\n          if (error.message) {\n            const matches = error.message.match(/index:[\\sa-zA-Z0-9_\\-\\.]+\\$?([a-zA-Z_-]+)_1/);\n            if (matches && Array.isArray(matches)) {\n              err.userInfo = { duplicated_field: matches[1] };\n            }\n          }\n          throw err;\n        }\n        throw error;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  // Remove all objects that match the given Parse Query.\n  // If no objects match, reject with OBJECT_NOT_FOUND. If objects are found and deleted, resolve with undefined.\n  // If there is some other error, reject with INTERNAL_SERVER_ERROR.\n  deleteObjectsByQuery(className: string, schema: SchemaType, query: QueryType) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    return this._adaptiveCollection(className)\n      .then(collection => {\n        const mongoWhere = transformWhere(className, query, schema);\n        return collection.deleteMany(mongoWhere)\n      })\n      .catch(err => this.handleError(err))\n      .then(({ result }) => {\n        if (result.n === 0) {\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Object not found.');\n        }\n        return Promise.resolve();\n      }, () => {\n        throw new Parse.Error(Parse.Error.INTERNAL_SERVER_ERROR, 'Database adapter error');\n      });\n  }\n\n  // Apply the update to all objects that match the given Parse Query.\n  updateObjectsByQuery(className: string, schema: SchemaType, query: QueryType, update: any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoUpdate = transformUpdate(className, update, schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.updateMany(mongoWhere, mongoUpdate))\n      .catch(err => this.handleError(err));\n  }\n\n  // Atomically finds and updates an object based on query.\n  // Return value not currently well specified.\n  findOneAndUpdate(className: string, schema: SchemaType, query: QueryType, update: any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoUpdate = transformUpdate(className, update, schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.findAndModify(mongoWhere, [], mongoUpdate, { new: true }))\n      .then(result => mongoObjectToParseObject(className, result.value, schema))\n      .catch(error => {\n        if (error.code === 11000) {\n          throw new Parse.Error(Parse.Error.DUPLICATE_VALUE, 'A duplicate value for a field with unique values was provided');\n        }\n        throw error;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  // Hopefully we can get rid of this. It's only used for config and hooks.\n  upsertOneObject(className: string, schema: SchemaType, query: QueryType, update: any) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoUpdate = transformUpdate(className, update, schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.upsertOne(mongoWhere, mongoUpdate))\n      .catch(err => this.handleError(err));\n  }\n\n  // Executes a find. Accepts: className, query in Parse format, and { skip, limit, sort }.\n  find(className: string, schema: SchemaType, query: QueryType, { skip, limit, sort, keys, readPreference }: QueryOptions): Promise<any> {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const mongoWhere = transformWhere(className, query, schema);\n    const mongoSort = _.mapKeys(sort, (value, fieldName) => transformKey(className, fieldName, schema));\n    const mongoKeys = _.reduce(keys, (memo, key) => {\n      memo[transformKey(className, key, schema)] = 1;\n      return memo;\n    }, {});\n\n    readPreference = this._parseReadPreference(readPreference);\n    return this.createTextIndexesIfNeeded(className, query, schema)\n      .then(() => this._adaptiveCollection(className))\n      .then(collection => collection.find(mongoWhere, {\n        skip,\n        limit,\n        sort: mongoSort,\n        keys: mongoKeys,\n        maxTimeMS: this._maxTimeMS,\n        readPreference,\n      }))\n      .then(objects => objects.map(object => mongoObjectToParseObject(className, object, schema)))\n      .catch(err => this.handleError(err));\n  }\n\n  // Create a unique index. Unique indexes on nullable fields are not allowed. Since we don't\n  // currently know which fields are nullable and which aren't, we ignore that criteria.\n  // As such, we shouldn't expose this function to users of parse until we have an out-of-band\n  // Way of determining if a field is nullable. Undefined doesn't count against uniqueness,\n  // which is why we use sparse indexes.\n  ensureUniqueness(className: string, schema: SchemaType, fieldNames: string[]) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const indexCreationRequest = {};\n    const mongoFieldNames = fieldNames.map(fieldName => transformKey(className, fieldName, schema));\n    mongoFieldNames.forEach(fieldName => {\n      indexCreationRequest[fieldName] = 1;\n    });\n    return this._adaptiveCollection(className)\n      .then(collection => collection._ensureSparseUniqueIndexInBackground(indexCreationRequest))\n      .catch(error => {\n        if (error.code === 11000) {\n          throw new Parse.Error(Parse.Error.DUPLICATE_VALUE, 'Tried to ensure field uniqueness for a class that already has duplicates.');\n        }\n        throw error;\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  // Used in tests\n  _rawFind(className: string, query: QueryType) {\n    return this._adaptiveCollection(className).then(collection => collection.find(query, {\n      maxTimeMS: this._maxTimeMS,\n    })).catch(err => this.handleError(err));\n  }\n\n  // Executes a count.\n  count(className: string, schema: SchemaType, query: QueryType, readPreference: ?string) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    readPreference = this._parseReadPreference(readPreference);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.count(transformWhere(className, query, schema), {\n        maxTimeMS: this._maxTimeMS,\n        readPreference,\n      }))\n      .catch(err => this.handleError(err));\n  }\n\n  distinct(className: string, schema: SchemaType, query: QueryType, fieldName: string) {\n    schema = convertParseSchemaToMongoSchema(schema);\n    const isPointerField = schema.fields[fieldName] && schema.fields[fieldName].type === 'Pointer';\n    if (isPointerField) {\n      fieldName = `_p_${fieldName}`\n    }\n    return this._adaptiveCollection(className)\n      .then(collection => collection.distinct(fieldName, transformWhere(className, query, schema)))\n      .then(objects => {\n        objects = objects.filter((obj) => obj != null);\n        return objects.map(object => {\n          if (isPointerField) {\n            const field = fieldName.substring(3);\n            return transformPointerString(schema, field, object);\n          }\n          return mongoObjectToParseObject(className, object, schema);\n        });\n      })\n      .catch(err => this.handleError(err));\n  }\n\n  aggregate(className: string, schema: any, pipeline: any, readPreference: ?string) {\n    let isPointerField = false;\n    pipeline = pipeline.map((stage) => {\n      if (stage.$group) {\n        stage.$group = this._parseAggregateGroupArgs(schema, stage.$group);\n        if (stage.$group._id && (typeof stage.$group._id === 'string') && stage.$group._id.indexOf('$_p_') >= 0) {\n          isPointerField = true;\n        }\n      }\n      if (stage.$match) {\n        stage.$match = this._parseAggregateArgs(schema, stage.$match);\n      }\n      if (stage.$project) {\n        stage.$project = this._parseAggregateProjectArgs(schema, stage.$project);\n      }\n      return stage;\n    });\n    readPreference = this._parseReadPreference(readPreference);\n    return this._adaptiveCollection(className)\n      .then(collection => collection.aggregate(pipeline, { readPreference, maxTimeMS: this._maxTimeMS }))\n      .catch(error => {\n        if (error.code === 16006) {\n          throw new Parse.Error(Parse.Error.INVALID_QUERY, error.message);\n        }\n        throw error;\n      })\n      .then(results => {\n        results.forEach(result => {\n          if (result.hasOwnProperty('_id')) {\n            if (isPointerField && result._id) {\n              result._id = result._id.split('$')[1];\n            }\n            if (result._id == null || _.isEmpty(result._id)) {\n              result._id = null;\n            }\n            result.objectId = result._id;\n            delete result._id;\n          }\n        });\n        return results;\n      })\n      .then(objects => objects.map(object => mongoObjectToParseObject(className, object, schema)))\n      .catch(err => this.handleError(err));\n  }\n\n  // This function will recursively traverse the pipeline and convert any Pointer or Date columns.\n  // If we detect a pointer column we will rename the column being queried for to match the column\n  // in the database. We also modify the value to what we expect the value to be in the database\n  // as well.\n  // For dates, the driver expects a Date object, but we have a string coming in. So we'll convert\n  // the string to a Date so the driver can perform the necessary comparison.\n  //\n  // The goal of this method is to look for the \"leaves\" of the pipeline and determine if it needs\n  // to be converted. The pipeline can have a few different forms. For more details, see:\n  //     https://docs.mongodb.com/manual/reference/operator/aggregation/\n  //\n  // If the pipeline is an array, it means we are probably parsing an '$and' or '$or' operator. In\n  // that case we need to loop through all of it's children to find the columns being operated on.\n  // If the pipeline is an object, then we'll loop through the keys checking to see if the key name\n  // matches one of the schema columns. If it does match a column and the column is a Pointer or\n  // a Date, then we'll convert the value as described above.\n  //\n  // As much as I hate recursion...this seemed like a good fit for it. We're essentially traversing\n  // down a tree to find a \"leaf node\" and checking to see if it needs to be converted.\n  _parseAggregateArgs(schema: any, pipeline: any): any {\n    if (Array.isArray(pipeline)) {\n      return pipeline.map((value) => this._parseAggregateArgs(schema, value));\n    } else if (typeof pipeline === 'object') {\n      const returnValue = {};\n      for (const field in pipeline) {\n        if (schema.fields[field] && schema.fields[field].type === 'Pointer') {\n          if (typeof pipeline[field] === 'object') {\n            // Pass objects down to MongoDB...this is more than likely an $exists operator.\n            returnValue[`_p_${field}`] = pipeline[field];\n          } else {\n            returnValue[`_p_${field}`] = `${schema.fields[field].targetClass}$${pipeline[field]}`;\n          }\n        } else if (schema.fields[field] && schema.fields[field].type === 'Date') {\n          returnValue[field] = this._convertToDate(pipeline[field]);\n        } else {\n          returnValue[field] = this._parseAggregateArgs(schema, pipeline[field]);\n        }\n\n        if (field === 'objectId') {\n          returnValue['_id'] = returnValue[field];\n          delete returnValue[field];\n        } else if (field === 'createdAt') {\n          returnValue['_created_at'] = returnValue[field];\n          delete returnValue[field];\n        } else if (field === 'updatedAt') {\n          returnValue['_updated_at'] = returnValue[field];\n          delete returnValue[field];\n        }\n      }\n      return returnValue;\n    }\n    return pipeline;\n  }\n\n  // This function is slightly different than the one above. Rather than trying to combine these\n  // two functions and making the code even harder to understand, I decided to split it up. The\n  // difference with this function is we are not transforming the values, only the keys of the\n  // pipeline.\n  _parseAggregateProjectArgs(schema: any, pipeline: any): any {\n    const returnValue = {};\n    for (const field in pipeline) {\n      if (schema.fields[field] && schema.fields[field].type === 'Pointer') {\n        returnValue[`_p_${field}`] = pipeline[field];\n      } else {\n        returnValue[field] = this._parseAggregateArgs(schema, pipeline[field]);\n      }\n\n      if (field === 'objectId') {\n        returnValue['_id'] = returnValue[field];\n        delete returnValue[field];\n      } else if (field === 'createdAt') {\n        returnValue['_created_at'] = returnValue[field];\n        delete returnValue[field];\n      } else if (field === 'updatedAt') {\n        returnValue['_updated_at'] = returnValue[field];\n        delete returnValue[field];\n      }\n    }\n    return returnValue;\n  }\n\n  // This function is slightly different than the two above. MongoDB $group aggregate looks like:\n  //     { $group: { _id: <expression>, <field1>: { <accumulator1> : <expression1> }, ... } }\n  // The <expression> could be a column name, prefixed with the '$' character. We'll look for\n  // these <expression> and check to see if it is a 'Pointer' or if it's one of createdAt,\n  // updatedAt or objectId and change it accordingly.\n  _parseAggregateGroupArgs(schema: any, pipeline: any): any {\n    if (Array.isArray(pipeline)) {\n      return pipeline.map((value) => this._parseAggregateGroupArgs(schema, value));\n    } else if (typeof pipeline === 'object') {\n      const returnValue = {};\n      for (const field in pipeline) {\n        returnValue[field] = this._parseAggregateGroupArgs(schema, pipeline[field]);\n      }\n      return returnValue;\n    } else if (typeof pipeline === 'string') {\n      const field = pipeline.substring(1);\n      if (schema.fields[field] && schema.fields[field].type === 'Pointer') {\n        return `$_p_${field}`;\n      } else if (field == 'createdAt') {\n        return '$_created_at';\n      } else if (field == 'updatedAt') {\n        return '$_updated_at';\n      }\n    }\n    return pipeline;\n  }\n\n  // This function will attempt to convert the provided value to a Date object. Since this is part\n  // of an aggregation pipeline, the value can either be a string or it can be another object with\n  // an operator in it (like $gt, $lt, etc). Because of this I felt it was easier to make this a\n  // recursive method to traverse down to the \"leaf node\" which is going to be the string.\n  _convertToDate(value: any): any {\n    if (typeof value === 'string') {\n      return new Date(value);\n    }\n\n    const returnValue = {}\n    for (const field in value) {\n      returnValue[field] = this._convertToDate(value[field])\n    }\n    return returnValue;\n  }\n\n  _parseReadPreference(readPreference: ?string): ?string {\n    switch (readPreference) {\n    case 'PRIMARY':\n      readPreference = ReadPreference.PRIMARY;\n      break;\n    case 'PRIMARY_PREFERRED':\n      readPreference = ReadPreference.PRIMARY_PREFERRED;\n      break;\n    case 'SECONDARY':\n      readPreference = ReadPreference.SECONDARY;\n      break;\n    case 'SECONDARY_PREFERRED':\n      readPreference = ReadPreference.SECONDARY_PREFERRED;\n      break;\n    case 'NEAREST':\n      readPreference = ReadPreference.NEAREST;\n      break;\n    case undefined:\n      break;\n    default:\n      throw new Parse.Error(Parse.Error.INVALID_QUERY, 'Not supported read preference.');\n    }\n    return readPreference;\n  }\n\n  performInitialization(): Promise<void> {\n    return Promise.resolve();\n  }\n\n  createIndex(className: string, index: any) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.createIndex(index, {background: true}))\n      .catch(err => this.handleError(err));\n  }\n\n  createIndexes(className: string, indexes: any) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.createIndexes(indexes, {background: true}))\n      .catch(err => this.handleError(err));\n  }\n\n  createIndexesIfNeeded(className: string, fieldName: string, type: any) {\n    if (type && type.type === 'Polygon') {\n      const index = {\n        [fieldName]: '2dsphere'\n      };\n      return this.createIndex(className, index);\n    }\n    return Promise.resolve();\n  }\n\n  createTextIndexesIfNeeded(className: string, query: QueryType, schema: any): Promise<void> {\n    for(const fieldName in query) {\n      if (!query[fieldName] || !query[fieldName].$text) {\n        continue;\n      }\n      const existingIndexes = schema.indexes;\n      for (const key in existingIndexes) {\n        const index = existingIndexes[key];\n        if (index.hasOwnProperty(fieldName)) {\n          return Promise.resolve();\n        }\n      }\n      const indexName = `${fieldName}_text`;\n      const textIndex = {\n        [indexName]: { [fieldName]: 'text' }\n      };\n      return this.setIndexesWithSchemaFormat(className, textIndex, existingIndexes, schema.fields)\n        .catch((error) => {\n          if (error.code === 85) { // Index exist with different options\n            return this.setIndexesFromMongo(className);\n          }\n          throw error;\n        });\n    }\n    return Promise.resolve();\n  }\n\n  getIndexes(className: string) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.indexes())\n      .catch(err => this.handleError(err));\n  }\n\n  dropIndex(className: string, index: any) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.dropIndex(index))\n      .catch(err => this.handleError(err));\n  }\n\n  dropAllIndexes(className: string) {\n    return this._adaptiveCollection(className)\n      .then(collection => collection._mongoCollection.dropIndexes())\n      .catch(err => this.handleError(err));\n  }\n\n  updateSchemaWithIndexes(): Promise<any> {\n    return this.getAllClasses()\n      .then((classes) => {\n        const promises = classes.map((schema) => {\n          return this.setIndexesFromMongo(schema.className);\n        });\n        return Promise.all(promises);\n      })\n      .catch(err => this.handleError(err));\n  }\n}\n\nexport default MongoStorageAdapter;\n"]} \ No newline at end of file diff --git a/lib/Controllers/PushController.js b/lib/Controllers/PushController.js index 9c783a58fd..1a65f7260d 100644 --- a/lib/Controllers/PushController.js +++ b/lib/Controllers/PushController.js @@ -21,6 +21,8 @@ var _StatusHandler = require('../StatusHandler'); var _utils = require('../Push/utils'); +var _logger = require('../logger'); + function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } class PushController { @@ -72,6 +74,8 @@ class PushController { badgeUpdate = () => { // Build a real RestQuery so we can use it in RestWrite const restQuery = new _RestQuery2.default(config, (0, _Auth.master)(config), '_Installation', updateWhere); + // change $exists for $ne null for better performance + if (restQuery.restWhere && restQuery.restWhere.deviceToken && restQuery.restWhere.deviceToken['$exists']) restQuery.restWhere.deviceToken = { $ne: null }; return restQuery.buildRestWhere().then(() => { const write = new _RestWrite2.default(config, (0, _Auth.master)(config), '_Installation', restQuery.restWhere, restUpdate); write.runOptions.many = true; @@ -84,7 +88,15 @@ class PushController { return pushStatus.setInitial(body, where); }).then(() => { onPushStatusSaved(pushStatus.objectId); - return badgeUpdate(); + const promise = badgeUpdate(); + // add this to ignore badge update errors as default + if (!config.stopOnBadgeUpdateError) { + promise.catch(err => { + _logger.logger.info(`Badge update error will be ignored for push status ${pushStatus.objectId}`); + _logger.logger.error(err); + }); + } + return promise; }).then(() => { // Update audience lastUsed and timesUsed if (body.audience_id) { @@ -212,4 +224,4 @@ class PushController { exports.PushController = PushController; exports.default = PushController; -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Controllers/PushController.js"],"names":["PushController","sendPush","body","where","config","auth","onPushStatusSaved","now","Date","hasPushSupport","Parse","Error","PUSH_MISCONFIGURED","expiration_time","getExpirationTime","expiration_interval","getExpirationInterval","hasOwnProperty","ttlMs","valueOf","pushTime","getPushTime","date","formatPushTime","badgeUpdate","Promise","resolve","data","badge","restUpdate","toLowerCase","__op","amount","Number","updateWhere","restQuery","RestQuery","buildRestWhere","then","write","RestWrite","restWhere","runOptions","many","execute","pushStatus","setInitial","objectId","audience_id","audienceId","updateAudience","lastUsed","__type","iso","toISOString","timesUsed","hasPushScheduledSupport","pushControllerQueue","enqueue","catch","err","fail","hasExpirationTime","expirationTimeParam","expirationTime","isFinite","hasExpirationInterval","expirationIntervalParam","hasPushTime","pushTimeParam","isLocalTime","pushTimeHasTimezoneComponent","offsetPattern","indexOf","length","test","isoString","substring"],"mappings":";;;;;;;AAAA;;AACA;;;;AACA;;;;AACA;;AACA;;AACA;;;;AAEO,MAAMA,cAAN,CAAqB;;AAE1BC,WAASC,OAAO,EAAhB,EAAoBC,QAAQ,EAA5B,EAAgCC,MAAhC,EAAwCC,IAAxC,EAA8CC,oBAAoB,MAAM,CAAE,CAA1E,EAA4EC,MAAM,IAAIC,IAAJ,EAAlF,EAA8F;AAC5F,QAAI,CAACJ,OAAOK,cAAZ,EAA4B;AAC1B,YAAM,IAAIC,YAAMC,KAAV,CAAgBD,YAAMC,KAAN,CAAYC,kBAA5B,EACJ,4BADI,CAAN;AAED;;AAED;AACAV,SAAKW,eAAL,GAAuBb,eAAec,iBAAf,CAAiCZ,IAAjC,CAAvB;AACAA,SAAKa,mBAAL,GAA2Bf,eAAegB,qBAAf,CAAqCd,IAArC,CAA3B;AACA,QAAIA,KAAKW,eAAL,IAAwBX,KAAKa,mBAAjC,EAAsD;AACpD,YAAM,IAAIL,YAAMC,KAAV,CACJD,YAAMC,KAAN,CAAYC,kBADR,EAEJ,4DAFI,CAAN;AAGD;;AAED;AACA,QAAIV,KAAKa,mBAAL,IAA4B,CAACb,KAAKe,cAAL,CAAoB,WAApB,CAAjC,EAAmE;AACjE,YAAMC,QAAQhB,KAAKa,mBAAL,GAA2B,IAAzC;AACAb,WAAKW,eAAL,GAAwB,IAAIL,IAAJ,CAASD,IAAIY,OAAJ,KAAgBD,KAAzB,CAAD,CAAkCC,OAAlC,EAAvB;AACD;;AAED,UAAMC,WAAWpB,eAAeqB,WAAf,CAA2BnB,IAA3B,CAAjB;AACA,QAAIkB,YAAYA,SAASE,IAAT,KAAkB,WAAlC,EAA+C;AAC7CpB,WAAK,WAAL,IAAoBF,eAAeuB,cAAf,CAA8BH,QAA9B,CAApB;AACD;;AAED;AACA;AACA,QAAII,cAAc,MAAM;AACtB,aAAOC,QAAQC,OAAR,EAAP;AACD,KAFD;;AAIA,QAAIxB,KAAKyB,IAAL,IAAazB,KAAKyB,IAAL,CAAUC,KAA3B,EAAkC;AAChC,YAAMA,QAAQ1B,KAAKyB,IAAL,CAAUC,KAAxB;AACA,UAAIC,aAAa,EAAjB;AACA,UAAI,OAAOD,KAAP,IAAgB,QAAhB,IAA4BA,MAAME,WAAN,OAAwB,WAAxD,EAAqE;AACnED,qBAAa,EAAED,OAAO,EAAEG,MAAM,WAAR,EAAqBC,QAAQ,CAA7B,EAAT,EAAb;AACD,OAFD,MAEO,IAAI,OAAOJ,KAAP,IAAgB,QAAhB,IAA4B,OAAOA,MAAMG,IAAb,IAAqB,QAAjD,IACAH,MAAMG,IAAN,CAAWD,WAAX,MAA4B,WAD5B,IAC2CG,OAAOL,MAAMI,MAAb,CAD/C,EACqE;AAC1EH,qBAAa,EAAED,OAAO,EAAEG,MAAM,WAAR,EAAqBC,QAAQJ,MAAMI,MAAnC,EAAT,EAAb;AACD,OAHM,MAGA,IAAIC,OAAOL,KAAP,CAAJ,EAAmB;AACxBC,qBAAa,EAAED,OAAOA,KAAT,EAAb;AACD,OAFM,MAEA;AACL,cAAM,gFAAN;AACD;;AAED;AACA,YAAMM,cAAc,mCAAuB/B,KAAvB,CAApB;AACAqB,oBAAc,MAAM;AAClB;AACA,cAAMW,YAAY,IAAIC,mBAAJ,CAAchC,MAAd,EAAsB,kBAAOA,MAAP,CAAtB,EAAsC,eAAtC,EAAuD8B,WAAvD,CAAlB;AACA,eAAOC,UAAUE,cAAV,GAA2BC,IAA3B,CAAgC,MAAM;AAC3C,gBAAMC,QAAQ,IAAIC,mBAAJ,CAAcpC,MAAd,EAAsB,kBAAOA,MAAP,CAAtB,EAAsC,eAAtC,EAAuD+B,UAAUM,SAAjE,EAA4EZ,UAA5E,CAAd;AACAU,gBAAMG,UAAN,CAAiBC,IAAjB,GAAwB,IAAxB;AACA,iBAAOJ,MAAMK,OAAN,EAAP;AACD,SAJM,CAAP;AAKD,OARD;AASD;AACD,UAAMC,aAAa,sCAAkBzC,MAAlB,CAAnB;AACA,WAAOqB,QAAQC,OAAR,GAAkBY,IAAlB,CAAuB,MAAM;AAClC,aAAOO,WAAWC,UAAX,CAAsB5C,IAAtB,EAA4BC,KAA5B,CAAP;AACD,KAFM,EAEJmC,IAFI,CAEC,MAAM;AACZhC,wBAAkBuC,WAAWE,QAA7B;AACA,aAAOvB,aAAP;AACD,KALM,EAKJc,IALI,CAKC,MAAM;AACZ;AACA,UAAIpC,KAAK8C,WAAT,EAAsB;AACpB,cAAMC,aAAa/C,KAAK8C,WAAxB;;AAEA,YAAIE,iBAAiB;AACnBC,oBAAU,EAAEC,QAAQ,MAAV,EAAkBC,KAAK,IAAI7C,IAAJ,GAAW8C,WAAX,EAAvB,EADS;AAEnBC,qBAAW,EAAExB,MAAM,WAAR,EAAqB,UAAU,CAA/B;AAFQ,SAArB;AAIA,cAAMQ,QAAQ,IAAIC,mBAAJ,CAAcpC,MAAd,EAAsB,kBAAOA,MAAP,CAAtB,EAAsC,WAAtC,EAAmD,EAAC2C,UAAUE,UAAX,EAAnD,EAA2EC,cAA3E,CAAd;AACAX,cAAMK,OAAN;AACD;AACD;AACA,aAAOnB,QAAQC,OAAR,EAAP;AACD,KAnBM,EAmBJY,IAnBI,CAmBC,MAAM;AACZ,UAAIpC,KAAKe,cAAL,CAAoB,WAApB,KAAoCb,OAAOoD,uBAA/C,EAAwE;AACtE,eAAO/B,QAAQC,OAAR,EAAP;AACD;AACD,aAAOtB,OAAOqD,mBAAP,CAA2BC,OAA3B,CAAmCxD,IAAnC,EAAyCC,KAAzC,EAAgDC,MAAhD,EAAwDC,IAAxD,EAA8DwC,UAA9D,CAAP;AACD,KAxBM,EAwBJc,KAxBI,CAwBGC,GAAD,IAAS;AAChB,aAAOf,WAAWgB,IAAX,CAAgBD,GAAhB,EAAqBtB,IAArB,CAA0B,MAAM;AACrC,cAAMsB,GAAN;AACD,OAFM,CAAP;AAGD,KA5BM,CAAP;AA6BD;;AAED;;;;;AAKA,SAAO9C,iBAAP,CAAyBZ,OAAO,EAAhC,EAAoC;AAClC,QAAI4D,oBAAoB5D,KAAKe,cAAL,CAAoB,iBAApB,CAAxB;AACA,QAAI,CAAC6C,iBAAL,EAAwB;AACtB;AACD;AACD,QAAIC,sBAAsB7D,KAAK,iBAAL,CAA1B;AACA,QAAI8D,cAAJ;AACA,QAAI,OAAOD,mBAAP,KAA+B,QAAnC,EAA6C;AAC3CC,uBAAiB,IAAIxD,IAAJ,CAASuD,sBAAsB,IAA/B,CAAjB;AACD,KAFD,MAEO,IAAI,OAAOA,mBAAP,KAA+B,QAAnC,EAA6C;AAClDC,uBAAiB,IAAIxD,IAAJ,CAASuD,mBAAT,CAAjB;AACD,KAFM,MAEA;AACL,YAAM,IAAIrD,YAAMC,KAAV,CAAgBD,YAAMC,KAAN,CAAYC,kBAA5B,EACJV,KAAK,iBAAL,IAA0B,qBADtB,CAAN;AAED;AACD;AACA,QAAI,CAAC+D,SAASD,cAAT,CAAL,EAA+B;AAC7B,YAAM,IAAItD,YAAMC,KAAV,CAAgBD,YAAMC,KAAN,CAAYC,kBAA5B,EACJV,KAAK,iBAAL,IAA0B,qBADtB,CAAN;AAED;AACD,WAAO8D,eAAe7C,OAAf,EAAP;AACD;;AAED,SAAOH,qBAAP,CAA6Bd,OAAO,EAApC,EAAwC;AACtC,UAAMgE,wBAAwBhE,KAAKe,cAAL,CAAoB,qBAApB,CAA9B;AACA,QAAI,CAACiD,qBAAL,EAA4B;AAC1B;AACD;;AAED,QAAIC,0BAA0BjE,KAAK,qBAAL,CAA9B;AACA,QAAI,OAAOiE,uBAAP,KAAmC,QAAnC,IAA+CA,2BAA2B,CAA9E,EAAiF;AAC/E,YAAM,IAAIzD,YAAMC,KAAV,CAAgBD,YAAMC,KAAN,CAAYC,kBAA5B,EACH,qDADG,CAAN;AAED;AACD,WAAOuD,uBAAP;AACD;;AAED;;;;;AAKA,SAAO9C,WAAP,CAAmBnB,OAAO,EAA1B,EAA8B;AAC5B,QAAIkE,cAAclE,KAAKe,cAAL,CAAoB,WAApB,CAAlB;AACA,QAAI,CAACmD,WAAL,EAAkB;AAChB;AACD;AACD,QAAIC,gBAAgBnE,KAAK,WAAL,CAApB;AACA,QAAIoB,IAAJ;AACA,QAAIgD,cAAc,IAAlB;;AAEA,QAAI,OAAOD,aAAP,KAAyB,QAA7B,EAAuC;AACrC/C,aAAO,IAAId,IAAJ,CAAS6D,gBAAgB,IAAzB,CAAP;AACD,KAFD,MAEO,IAAI,OAAOA,aAAP,KAAyB,QAA7B,EAAuC;AAC5CC,oBAAc,CAACtE,eAAeuE,4BAAf,CAA4CF,aAA5C,CAAf;AACA/C,aAAO,IAAId,IAAJ,CAAS6D,aAAT,CAAP;AACD,KAHM,MAGA;AACL,YAAM,IAAI3D,YAAMC,KAAV,CAAgBD,YAAMC,KAAN,CAAYC,kBAA5B,EACJV,KAAK,WAAL,IAAoB,qBADhB,CAAN;AAED;AACD;AACA,QAAI,CAAC+D,SAAS3C,IAAT,CAAL,EAAqB;AACnB,YAAM,IAAIZ,YAAMC,KAAV,CAAgBD,YAAMC,KAAN,CAAYC,kBAA5B,EACJV,KAAK,WAAL,IAAoB,qBADhB,CAAN;AAED;;AAED,WAAO;AACLoB,UADK;AAELgD;AAFK,KAAP;AAID;;AAED;;;;;AAKA,SAAOC,4BAAP,CAAoCF,aAApC,EAAoE;AAClE,UAAMG,gBAAgB,sBAAtB;AACA,WAAOH,cAAcI,OAAd,CAAsB,GAAtB,MAA+BJ,cAAcK,MAAd,GAAuB,CAAtD,CAAwD;AAAxD,OACFF,cAAcG,IAAd,CAAmBN,aAAnB,CADL,CAFkE,CAG1B;AACzC;;AAED;;;;;;AAMA,SAAO9C,cAAP,CAAsB,EAAED,IAAF,EAAQgD,WAAR,EAAtB,EAAmF;AACjF,QAAIA,WAAJ,EAAiB;AAAE;AACjB,YAAMM,YAAYtD,KAAKgC,WAAL,EAAlB;AACA,aAAOsB,UAAUC,SAAV,CAAoB,CAApB,EAAuBD,UAAUH,OAAV,CAAkB,GAAlB,CAAvB,CAAP;AACD;AACD,WAAOnD,KAAKgC,WAAL,EAAP;AACD;AAhMyB;;QAAftD,c,GAAAA,c;kBAmMEA,c","file":"PushController.js","sourcesContent":["import { Parse }              from 'parse/node';\nimport RestQuery              from '../RestQuery';\nimport RestWrite              from '../RestWrite';\nimport { master }             from '../Auth';\nimport { pushStatusHandler }  from '../StatusHandler';\nimport { applyDeviceTokenExists } from '../Push/utils';\n\nexport class PushController {\n\n  sendPush(body = {}, where = {}, config, auth, onPushStatusSaved = () => {}, now = new Date()) {\n    if (!config.hasPushSupport) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        'Missing push configuration');\n    }\n\n    // Replace the expiration_time and push_time with a valid Unix epoch milliseconds time\n    body.expiration_time = PushController.getExpirationTime(body);\n    body.expiration_interval = PushController.getExpirationInterval(body);\n    if (body.expiration_time && body.expiration_interval) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        'Both expiration_time and expiration_interval cannot be set');\n    }\n\n    // Immediate push\n    if (body.expiration_interval && !body.hasOwnProperty('push_time')) {\n      const ttlMs = body.expiration_interval * 1000;\n      body.expiration_time = (new Date(now.valueOf() + ttlMs)).valueOf();\n    }\n\n    const pushTime = PushController.getPushTime(body);\n    if (pushTime && pushTime.date !== 'undefined') {\n      body['push_time'] = PushController.formatPushTime(pushTime);\n    }\n\n    // TODO: If the req can pass the checking, we return immediately instead of waiting\n    // pushes to be sent. We probably change this behaviour in the future.\n    let badgeUpdate = () => {\n      return Promise.resolve();\n    }\n\n    if (body.data && body.data.badge) {\n      const badge = body.data.badge;\n      let restUpdate = {};\n      if (typeof badge == 'string' && badge.toLowerCase() === 'increment') {\n        restUpdate = { badge: { __op: 'Increment', amount: 1 } }\n      } else if (typeof badge == 'object' && typeof badge.__op == 'string' &&\n                 badge.__op.toLowerCase() == 'increment' && Number(badge.amount)) {\n        restUpdate = { badge: { __op: 'Increment', amount: badge.amount } }\n      } else if (Number(badge)) {\n        restUpdate = { badge: badge }\n      } else {\n        throw \"Invalid value for badge, expected number or 'Increment' or {increment: number}\";\n      }\n\n      // Force filtering on only valid device tokens\n      const updateWhere = applyDeviceTokenExists(where);\n      badgeUpdate = () => {\n        // Build a real RestQuery so we can use it in RestWrite\n        const restQuery = new RestQuery(config, master(config), '_Installation', updateWhere);\n        return restQuery.buildRestWhere().then(() => {\n          const write = new RestWrite(config, master(config), '_Installation', restQuery.restWhere, restUpdate);\n          write.runOptions.many = true;\n          return write.execute();\n        });\n      }\n    }\n    const pushStatus = pushStatusHandler(config);\n    return Promise.resolve().then(() => {\n      return pushStatus.setInitial(body, where);\n    }).then(() => {\n      onPushStatusSaved(pushStatus.objectId);\n      return badgeUpdate();\n    }).then(() => {\n      // Update audience lastUsed and timesUsed\n      if (body.audience_id) {\n        const audienceId = body.audience_id;\n\n        var updateAudience = {\n          lastUsed: { __type: \"Date\", iso: new Date().toISOString() },\n          timesUsed: { __op: \"Increment\", \"amount\": 1 }\n        };\n        const write = new RestWrite(config, master(config), '_Audience', {objectId: audienceId}, updateAudience);\n        write.execute();\n      }\n      // Don't wait for the audience update promise to resolve.\n      return Promise.resolve();\n    }).then(() => {\n      if (body.hasOwnProperty('push_time') && config.hasPushScheduledSupport) {\n        return Promise.resolve();\n      }\n      return config.pushControllerQueue.enqueue(body, where, config, auth, pushStatus);\n    }).catch((err) => {\n      return pushStatus.fail(err).then(() => {\n        throw err;\n      });\n    });\n  }\n\n  /**\n   * Get expiration time from the request body.\n   * @param {Object} request A request object\n   * @returns {Number|undefined} The expiration time if it exists in the request\n   */\n  static getExpirationTime(body = {}) {\n    var hasExpirationTime = body.hasOwnProperty('expiration_time');\n    if (!hasExpirationTime) {\n      return;\n    }\n    var expirationTimeParam = body['expiration_time'];\n    var expirationTime;\n    if (typeof expirationTimeParam === 'number') {\n      expirationTime = new Date(expirationTimeParam * 1000);\n    } else if (typeof expirationTimeParam === 'string') {\n      expirationTime = new Date(expirationTimeParam);\n    } else {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        body['expiration_time'] + ' is not valid time.');\n    }\n    // Check expirationTime is valid or not, if it is not valid, expirationTime is NaN\n    if (!isFinite(expirationTime)) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        body['expiration_time'] + ' is not valid time.');\n    }\n    return expirationTime.valueOf();\n  }\n\n  static getExpirationInterval(body = {}) {\n    const hasExpirationInterval = body.hasOwnProperty('expiration_interval');\n    if (!hasExpirationInterval) {\n      return;\n    }\n\n    var expirationIntervalParam = body['expiration_interval'];\n    if (typeof expirationIntervalParam !== 'number' || expirationIntervalParam <= 0) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        `expiration_interval must be a number greater than 0`);\n    }\n    return expirationIntervalParam;\n  }\n\n  /**\n   * Get push time from the request body.\n   * @param {Object} request A request object\n   * @returns {Number|undefined} The push time if it exists in the request\n   */\n  static getPushTime(body = {}) {\n    var hasPushTime = body.hasOwnProperty('push_time');\n    if (!hasPushTime) {\n      return;\n    }\n    var pushTimeParam = body['push_time'];\n    var date;\n    var isLocalTime = true;\n\n    if (typeof pushTimeParam === 'number') {\n      date = new Date(pushTimeParam * 1000);\n    } else if (typeof pushTimeParam === 'string') {\n      isLocalTime = !PushController.pushTimeHasTimezoneComponent(pushTimeParam);\n      date = new Date(pushTimeParam);\n    } else {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        body['push_time'] + ' is not valid time.');\n    }\n    // Check pushTime is valid or not, if it is not valid, pushTime is NaN\n    if (!isFinite(date)) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        body['push_time'] + ' is not valid time.');\n    }\n\n    return {\n      date,\n      isLocalTime,\n    };\n  }\n\n  /**\n   * Checks if a ISO8601 formatted date contains a timezone component\n   * @param pushTimeParam {string}\n   * @returns {boolean}\n   */\n  static pushTimeHasTimezoneComponent(pushTimeParam: string): boolean {\n    const offsetPattern = /(.+)([+-])\\d\\d:\\d\\d$/;\n    return pushTimeParam.indexOf('Z') === pushTimeParam.length - 1 // 2007-04-05T12:30Z\n      || offsetPattern.test(pushTimeParam); // 2007-04-05T12:30.000+02:00, 2007-04-05T12:30.000-02:00\n  }\n\n  /**\n   * Converts a date to ISO format in UTC time and strips the timezone if `isLocalTime` is true\n   * @param date {Date}\n   * @param isLocalTime {boolean}\n   * @returns {string}\n   */\n  static formatPushTime({ date, isLocalTime }: { date: Date, isLocalTime: boolean }) {\n    if (isLocalTime) { // Strip 'Z'\n      const isoString = date.toISOString();\n      return isoString.substring(0, isoString.indexOf('Z'));\n    }\n    return date.toISOString();\n  }\n}\n\nexport default PushController;\n"]} \ No newline at end of file +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Controllers/PushController.js"],"names":["PushController","sendPush","body","where","config","auth","onPushStatusSaved","now","Date","hasPushSupport","Parse","Error","PUSH_MISCONFIGURED","expiration_time","getExpirationTime","expiration_interval","getExpirationInterval","hasOwnProperty","ttlMs","valueOf","pushTime","getPushTime","date","formatPushTime","badgeUpdate","Promise","resolve","data","badge","restUpdate","toLowerCase","__op","amount","Number","updateWhere","restQuery","RestQuery","restWhere","deviceToken","$ne","buildRestWhere","then","write","RestWrite","runOptions","many","execute","pushStatus","setInitial","objectId","promise","stopOnBadgeUpdateError","catch","err","logger","info","error","audience_id","audienceId","updateAudience","lastUsed","__type","iso","toISOString","timesUsed","hasPushScheduledSupport","pushControllerQueue","enqueue","fail","hasExpirationTime","expirationTimeParam","expirationTime","isFinite","hasExpirationInterval","expirationIntervalParam","hasPushTime","pushTimeParam","isLocalTime","pushTimeHasTimezoneComponent","offsetPattern","indexOf","length","test","isoString","substring"],"mappings":"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","file":"PushController.js","sourcesContent":["import { Parse }              from 'parse/node';\nimport RestQuery              from '../RestQuery';\nimport RestWrite              from '../RestWrite';\nimport { master }             from '../Auth';\nimport { pushStatusHandler }  from '../StatusHandler';\nimport { applyDeviceTokenExists } from '../Push/utils';\nimport { logger }               from '../logger';\n\nexport class PushController {\n\n  sendPush(body = {}, where = {}, config, auth, onPushStatusSaved = () => {}, now = new Date()) {\n    if (!config.hasPushSupport) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        'Missing push configuration');\n    }\n\n    // Replace the expiration_time and push_time with a valid Unix epoch milliseconds time\n    body.expiration_time = PushController.getExpirationTime(body);\n    body.expiration_interval = PushController.getExpirationInterval(body);\n    if (body.expiration_time && body.expiration_interval) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        'Both expiration_time and expiration_interval cannot be set');\n    }\n\n    // Immediate push\n    if (body.expiration_interval && !body.hasOwnProperty('push_time')) {\n      const ttlMs = body.expiration_interval * 1000;\n      body.expiration_time = (new Date(now.valueOf() + ttlMs)).valueOf();\n    }\n\n    const pushTime = PushController.getPushTime(body);\n    if (pushTime && pushTime.date !== 'undefined') {\n      body['push_time'] = PushController.formatPushTime(pushTime);\n    }\n\n    // TODO: If the req can pass the checking, we return immediately instead of waiting\n    // pushes to be sent. We probably change this behaviour in the future.\n    let badgeUpdate = () => {\n      return Promise.resolve();\n    }\n\n    if (body.data && body.data.badge) {\n      const badge = body.data.badge;\n      let restUpdate = {};\n      if (typeof badge == 'string' && badge.toLowerCase() === 'increment') {\n        restUpdate = { badge: { __op: 'Increment', amount: 1 } }\n      } else if (typeof badge == 'object' && typeof badge.__op == 'string' &&\n                 badge.__op.toLowerCase() == 'increment' && Number(badge.amount)) {\n        restUpdate = { badge: { __op: 'Increment', amount: badge.amount } }\n      } else if (Number(badge)) {\n        restUpdate = { badge: badge }\n      } else {\n        throw \"Invalid value for badge, expected number or 'Increment' or {increment: number}\";\n      }\n\n      // Force filtering on only valid device tokens\n      const updateWhere = applyDeviceTokenExists(where);\n      badgeUpdate = () => {\n        // Build a real RestQuery so we can use it in RestWrite\n        const restQuery = new RestQuery(config, master(config), '_Installation', updateWhere);\n        // change $exists for $ne null for better performance\n        if (restQuery.restWhere && restQuery.restWhere.deviceToken && restQuery.restWhere.deviceToken['$exists']) restQuery.restWhere.deviceToken = {$ne: null}\n        return restQuery.buildRestWhere().then(() => {\n          const write = new RestWrite(config, master(config), '_Installation', restQuery.restWhere, restUpdate);\n          write.runOptions.many = true;\n          return write.execute();\n        });\n      }\n    }\n    const pushStatus = pushStatusHandler(config);\n    return Promise.resolve().then(() => {\n      return pushStatus.setInitial(body, where);\n    }).then(() => {\n      onPushStatusSaved(pushStatus.objectId);\n      const promise = badgeUpdate();\n      // add this to ignore badge update errors as default\n      if (!config.stopOnBadgeUpdateError) {\n        promise.catch(err => {\n          logger.info(`Badge update error will be ignored for push status ${pushStatus.objectId}`)\n          logger.error(err)\n        })\n      }\n      return promise\n    }).then(() => {\n      // Update audience lastUsed and timesUsed\n      if (body.audience_id) {\n        const audienceId = body.audience_id;\n\n        var updateAudience = {\n          lastUsed: { __type: \"Date\", iso: new Date().toISOString() },\n          timesUsed: { __op: \"Increment\", \"amount\": 1 }\n        };\n        const write = new RestWrite(config, master(config), '_Audience', {objectId: audienceId}, updateAudience);\n        write.execute();\n      }\n      // Don't wait for the audience update promise to resolve.\n      return Promise.resolve();\n    }).then(() => {\n      if (body.hasOwnProperty('push_time') && config.hasPushScheduledSupport) {\n        return Promise.resolve();\n      }\n      return config.pushControllerQueue.enqueue(body, where, config, auth, pushStatus);\n    }).catch((err) => {\n      return pushStatus.fail(err).then(() => {\n        throw err;\n      });\n    });\n  }\n\n  /**\n   * Get expiration time from the request body.\n   * @param {Object} request A request object\n   * @returns {Number|undefined} The expiration time if it exists in the request\n   */\n  static getExpirationTime(body = {}) {\n    var hasExpirationTime = body.hasOwnProperty('expiration_time');\n    if (!hasExpirationTime) {\n      return;\n    }\n    var expirationTimeParam = body['expiration_time'];\n    var expirationTime;\n    if (typeof expirationTimeParam === 'number') {\n      expirationTime = new Date(expirationTimeParam * 1000);\n    } else if (typeof expirationTimeParam === 'string') {\n      expirationTime = new Date(expirationTimeParam);\n    } else {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        body['expiration_time'] + ' is not valid time.');\n    }\n    // Check expirationTime is valid or not, if it is not valid, expirationTime is NaN\n    if (!isFinite(expirationTime)) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        body['expiration_time'] + ' is not valid time.');\n    }\n    return expirationTime.valueOf();\n  }\n\n  static getExpirationInterval(body = {}) {\n    const hasExpirationInterval = body.hasOwnProperty('expiration_interval');\n    if (!hasExpirationInterval) {\n      return;\n    }\n\n    var expirationIntervalParam = body['expiration_interval'];\n    if (typeof expirationIntervalParam !== 'number' || expirationIntervalParam <= 0) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        `expiration_interval must be a number greater than 0`);\n    }\n    return expirationIntervalParam;\n  }\n\n  /**\n   * Get push time from the request body.\n   * @param {Object} request A request object\n   * @returns {Number|undefined} The push time if it exists in the request\n   */\n  static getPushTime(body = {}) {\n    var hasPushTime = body.hasOwnProperty('push_time');\n    if (!hasPushTime) {\n      return;\n    }\n    var pushTimeParam = body['push_time'];\n    var date;\n    var isLocalTime = true;\n\n    if (typeof pushTimeParam === 'number') {\n      date = new Date(pushTimeParam * 1000);\n    } else if (typeof pushTimeParam === 'string') {\n      isLocalTime = !PushController.pushTimeHasTimezoneComponent(pushTimeParam);\n      date = new Date(pushTimeParam);\n    } else {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        body['push_time'] + ' is not valid time.');\n    }\n    // Check pushTime is valid or not, if it is not valid, pushTime is NaN\n    if (!isFinite(date)) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED,\n        body['push_time'] + ' is not valid time.');\n    }\n\n    return {\n      date,\n      isLocalTime,\n    };\n  }\n\n  /**\n   * Checks if a ISO8601 formatted date contains a timezone component\n   * @param pushTimeParam {string}\n   * @returns {boolean}\n   */\n  static pushTimeHasTimezoneComponent(pushTimeParam: string): boolean {\n    const offsetPattern = /(.+)([+-])\\d\\d:\\d\\d$/;\n    return pushTimeParam.indexOf('Z') === pushTimeParam.length - 1 // 2007-04-05T12:30Z\n      || offsetPattern.test(pushTimeParam); // 2007-04-05T12:30.000+02:00, 2007-04-05T12:30.000-02:00\n  }\n\n  /**\n   * Converts a date to ISO format in UTC time and strips the timezone if `isLocalTime` is true\n   * @param date {Date}\n   * @param isLocalTime {boolean}\n   * @returns {string}\n   */\n  static formatPushTime({ date, isLocalTime }: { date: Date, isLocalTime: boolean }) {\n    if (isLocalTime) { // Strip 'Z'\n      const isoString = date.toISOString();\n      return isoString.substring(0, isoString.indexOf('Z'));\n    }\n    return date.toISOString();\n  }\n}\n\nexport default PushController;\n"]} \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index ca05b338ff..861010fdf7 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1231,25 +1231,25 @@ }, "babel-plugin-syntax-async-functions": { "version": "6.13.0", - "resolved": "https://registry.npmjs.org/babel-plugin-syntax-async-functions/-/babel-plugin-syntax-async-functions-6.13.0.tgz", + "resolved": "http://registry.npmjs.org/babel-plugin-syntax-async-functions/-/babel-plugin-syntax-async-functions-6.13.0.tgz", "integrity": "sha1-ytnK0RkbWtY0vzCuCHI5HgZHvpU=", "dev": true }, "babel-plugin-syntax-exponentiation-operator": { "version": "6.13.0", - "resolved": "https://registry.npmjs.org/babel-plugin-syntax-exponentiation-operator/-/babel-plugin-syntax-exponentiation-operator-6.13.0.tgz", + "resolved": "http://registry.npmjs.org/babel-plugin-syntax-exponentiation-operator/-/babel-plugin-syntax-exponentiation-operator-6.13.0.tgz", "integrity": "sha1-nufoM3KQ2pUoggGmpX9BcDF4MN4=", "dev": true }, "babel-plugin-syntax-flow": { "version": "6.18.0", - "resolved": "https://registry.npmjs.org/babel-plugin-syntax-flow/-/babel-plugin-syntax-flow-6.18.0.tgz", + "resolved": "http://registry.npmjs.org/babel-plugin-syntax-flow/-/babel-plugin-syntax-flow-6.18.0.tgz", "integrity": "sha1-TDqyCiryaqIM0lmVw5jE63AxDI0=", "dev": true }, "babel-plugin-syntax-object-rest-spread": { "version": "6.13.0", - "resolved": "https://registry.npmjs.org/babel-plugin-syntax-object-rest-spread/-/babel-plugin-syntax-object-rest-spread-6.13.0.tgz", + "resolved": "http://registry.npmjs.org/babel-plugin-syntax-object-rest-spread/-/babel-plugin-syntax-object-rest-spread-6.13.0.tgz", "integrity": "sha1-/WU28rzhODb/o6VFjEkDpZe7O/U=", "dev": true }, @@ -5531,7 +5531,7 @@ "http-proxy-agent": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-2.1.0.tgz", - "integrity": "sha1-5IIb7vWyFCogJr1zkm/lN2McVAU=", + "integrity": "sha512-qwHbBLV7WviBl0rQsOzH6o5lwyOIvwp/BdFnvVxXORldu5TmjFfjzBcWUWS5kWAZhmv+JtiDhSuQCp4sBfbIgg==", "requires": { "agent-base": "4.2.0", "debug": "3.1.0" @@ -5550,7 +5550,7 @@ "https-proxy-agent": { "version": "2.2.1", "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-2.2.1.tgz", - "integrity": "sha1-UVUpcPoE1yPgTFbQQXjD+SWSu8A=", + "integrity": "sha512-HPCTS1LW51bcyMYbxUIOO4HEOlQ1/1qRaFWcyxvwaqUS9TY88aoEuHUY33kuAh1YhVVaDQhLZsnPd+XNARWZlQ==", "requires": { "agent-base": "4.2.0", "debug": "3.1.0" @@ -6709,7 +6709,7 @@ "mailgun-js": { "version": "0.18.0", "resolved": "https://registry.npmjs.org/mailgun-js/-/mailgun-js-0.18.0.tgz", - "integrity": "sha1-gf7QxmpBHT/2xDVIYa0hOHr8+qo=", + "integrity": "sha512-o0P6jjZlx5CQj12tvVgDTbgjTqVN0+5h6/6P1+3c6xmozVKBwniQ6Qt3MkCSF0+ueVTbobAfWyGpWRZMJu8t1g==", "requires": { "async": "2.6.0", "debug": "3.1.0", From 450192ca0ac7c9fe2b43838ff43b79ab21496596 Mon Sep 17 00:00:00 2001 From: Ricardo Paiva Date: Wed, 26 Sep 2018 13:03:56 -0300 Subject: [PATCH 3/3] Fix ignore badge update error --- src/Controllers/PushController.js | 13 +++++-------- 1 file changed, 5 insertions(+), 8 deletions(-) diff --git a/src/Controllers/PushController.js b/src/Controllers/PushController.js index f58fb67ad6..77829d216e 100644 --- a/src/Controllers/PushController.js +++ b/src/Controllers/PushController.js @@ -73,15 +73,12 @@ export class PushController { return pushStatus.setInitial(body, where); }).then(() => { onPushStatusSaved(pushStatus.objectId); - const promise = badgeUpdate(); + return badgeUpdate(); + }).catch(err => { // add this to ignore badge update errors as default - if (!config.stopOnBadgeUpdateError) { - promise.catch(err => { - logger.info(`Badge update error will be ignored for push status ${pushStatus.objectId}`) - logger.error(err) - }) - } - return promise + if (config.stopOnBadgeUpdateError) throw err + logger.info(`Badge update error will be ignored for push status ${pushStatus.objectId}`) + logger.error(err) }).then(() => { // Update audience lastUsed and timesUsed if (body.audience_id) {