parse-server 5.4.0-beta.1 → 6.0.0-alpha.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +6 -6
- package/lib/Adapters/Auth/AuthAdapter.js +92 -11
- package/lib/Adapters/Auth/index.js +81 -23
- package/lib/Adapters/Cache/LRUCache.js +3 -3
- package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +7 -2
- package/lib/Auth.js +214 -6
- package/lib/Config.js +10 -2
- package/lib/Controllers/DatabaseController.js +4 -3
- package/lib/Deprecator/Deprecations.js +4 -1
- package/lib/GraphQL/loaders/parseClassTypes.js +9 -4
- package/lib/GraphQL/loaders/usersMutations.js +75 -9
- package/lib/Options/Definitions.js +7 -1
- package/lib/Options/docs.js +2 -1
- package/lib/Options/index.js +1 -1
- package/lib/RestQuery.js +3 -3
- package/lib/RestWrite.js +165 -166
- package/lib/Routers/FilesRouter.js +16 -1
- package/lib/Routers/UsersRouter.js +167 -3
- package/lib/cloud-code/Parse.Cloud.js +2 -1
- package/package.json +5 -5
|
@@ -29,6 +29,8 @@ const triggers = require('../triggers');
|
|
|
29
29
|
|
|
30
30
|
const http = require('http');
|
|
31
31
|
|
|
32
|
+
const Utils = require('../Utils');
|
|
33
|
+
|
|
32
34
|
const downloadFileFromURI = uri => {
|
|
33
35
|
return new Promise((res, rej) => {
|
|
34
36
|
http.get(uri, response => {
|
|
@@ -161,6 +163,19 @@ class FilesRouter {
|
|
|
161
163
|
metadata = {},
|
|
162
164
|
tags = {}
|
|
163
165
|
} = req.fileData || {};
|
|
166
|
+
|
|
167
|
+
if (req.config && req.config.requestKeywordDenylist) {
|
|
168
|
+
// Scan request data for denied keywords
|
|
169
|
+
for (const keyword of req.config.requestKeywordDenylist) {
|
|
170
|
+
const match = Utils.objectContainsKeyValue(metadata, keyword.key, keyword.value) || Utils.objectContainsKeyValue(tags, keyword.key, keyword.value);
|
|
171
|
+
|
|
172
|
+
if (match) {
|
|
173
|
+
next(new _node.default.Error(_node.default.Error.INVALID_KEY_NAME, `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`));
|
|
174
|
+
return;
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
|
|
164
179
|
file.setTags(tags);
|
|
165
180
|
file.setMetadata(metadata);
|
|
166
181
|
const fileSize = Buffer.byteLength(req.body);
|
|
@@ -296,4 +311,4 @@ function isFileStreamable(req, filesController) {
|
|
|
296
311
|
const end = Number(range[1]);
|
|
297
312
|
return (!isNaN(start) || !isNaN(end)) && typeof filesController.adapter.handleFileStream === 'function';
|
|
298
313
|
}
|
|
299
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Routers/FilesRouter.js"],"names":["triggers","require","http","downloadFileFromURI","uri","Promise","res","rej","get","response","setDefaultEncoding","body","headers","on","data","e","message","addFileDataIfNeeded","file","_source","format","base64","_previousSave","_data","_requestTask","FilesRouter","expressRouter","maxUploadSize","router","express","Router","getHandler","metadataHandler","post","req","next","Parse","Error","INVALID_FILE_NAME","BodyParser","raw","type","limit","Middlewares","handleParseHeaders","createHandler","delete","enforceMasterKeyAccess","deleteHandler","config","Config","params","appId","status","err","OPERATION_FORBIDDEN","json","code","error","filesController","filename","contentType","mime","getType","isFileStreamable","handleFileStream","catch","set","end","getFileData","then","length","user","auth","isMaster","isLinked","AnonymousUtils","fileUpload","enableForAnonymousUser","FILE_SAVE_ERROR","enableForAuthenticatedUser","enableForPublic","validateFilename","toString","File","metadata","tags","fileData","setTags","setMetadata","fileSize","Buffer","byteLength","fileObject","triggerResult","maybeRunFileTrigger","Types","beforeSave","saveResult","url","name","_name","bufferData","from","fileOptions","_metadata","fileTags","Object","keys","_tags","assign","createFileResult","createFile","_url","resolve","afterSave","logger","resolveError","adapter","getFileLocation","beforeDelete","deleteFile","afterDelete","FILE_DELETE_ERROR","getMetadata","range","split","start","Number","isNaN"],"mappings":"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","sourcesContent":["import express from 'express';\nimport BodyParser from 'body-parser';\nimport * as Middlewares from '../middlewares';\nimport Parse from 'parse/node';\nimport Config from '../Config';\nimport mime from 'mime';\nimport logger from '../logger';\nconst triggers = require('../triggers');\nconst http = require('http');\n\nconst downloadFileFromURI = uri => {\n  return new Promise((res, rej) => {\n    http\n      .get(uri, response => {\n        response.setDefaultEncoding('base64');\n        let body = `data:${response.headers['content-type']};base64,`;\n        response.on('data', data => (body += data));\n        response.on('end', () => res(body));\n      })\n      .on('error', e => {\n        rej(`Error downloading file from ${uri}: ${e.message}`);\n      });\n  });\n};\n\nconst addFileDataIfNeeded = async file => {\n  if (file._source.format === 'uri') {\n    const base64 = await downloadFileFromURI(file._source.uri);\n    file._previousSave = file;\n    file._data = base64;\n    file._requestTask = null;\n  }\n  return file;\n};\n\nexport class FilesRouter {\n  expressRouter({ maxUploadSize = '20Mb' } = {}) {\n    var router = express.Router();\n    router.get('/files/:appId/:filename', this.getHandler);\n    router.get('/files/:appId/metadata/:filename', this.metadataHandler);\n\n    router.post('/files', function (req, res, next) {\n      next(new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Filename not provided.'));\n    });\n\n    router.post(\n      '/files/:filename',\n      BodyParser.raw({\n        type: () => {\n          return true;\n        },\n        limit: maxUploadSize,\n      }), // Allow uploads without Content-Type, or with any Content-Type.\n      Middlewares.handleParseHeaders,\n      this.createHandler\n    );\n\n    router.delete(\n      '/files/:filename',\n      Middlewares.handleParseHeaders,\n      Middlewares.enforceMasterKeyAccess,\n      this.deleteHandler\n    );\n    return router;\n  }\n\n  getHandler(req, res) {\n    const config = Config.get(req.params.appId);\n    if (!config) {\n      res.status(403);\n      const err = new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, 'Invalid application ID.');\n      res.json({ code: err.code, error: err.message });\n      return;\n    }\n    const filesController = config.filesController;\n    const filename = req.params.filename;\n    const contentType = mime.getType(filename);\n    if (isFileStreamable(req, filesController)) {\n      filesController.handleFileStream(config, filename, req, res, contentType).catch(() => {\n        res.status(404);\n        res.set('Content-Type', 'text/plain');\n        res.end('File not found.');\n      });\n    } else {\n      filesController\n        .getFileData(config, filename)\n        .then(data => {\n          res.status(200);\n          res.set('Content-Type', contentType);\n          res.set('Content-Length', data.length);\n          res.end(data);\n        })\n        .catch(() => {\n          res.status(404);\n          res.set('Content-Type', 'text/plain');\n          res.end('File not found.');\n        });\n    }\n  }\n\n  async createHandler(req, res, next) {\n    const config = req.config;\n    const user = req.auth.user;\n    const isMaster = req.auth.isMaster;\n    const isLinked = user && Parse.AnonymousUtils.isLinked(user);\n    if (!isMaster && !config.fileUpload.enableForAnonymousUser && isLinked) {\n      next(\n        new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'File upload by anonymous user is disabled.')\n      );\n      return;\n    }\n    if (!isMaster && !config.fileUpload.enableForAuthenticatedUser && !isLinked && user) {\n      next(\n        new Parse.Error(\n          Parse.Error.FILE_SAVE_ERROR,\n          'File upload by authenticated user is disabled.'\n        )\n      );\n      return;\n    }\n    if (!isMaster && !config.fileUpload.enableForPublic && !user) {\n      next(new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'File upload by public is disabled.'));\n      return;\n    }\n    const filesController = config.filesController;\n    const { filename } = req.params;\n    const contentType = req.get('Content-type');\n\n    if (!req.body || !req.body.length) {\n      next(new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'Invalid file upload.'));\n      return;\n    }\n\n    const error = filesController.validateFilename(filename);\n    if (error) {\n      next(error);\n      return;\n    }\n\n    const base64 = req.body.toString('base64');\n    const file = new Parse.File(filename, { base64 }, contentType);\n    const { metadata = {}, tags = {} } = req.fileData || {};\n    file.setTags(tags);\n    file.setMetadata(metadata);\n    const fileSize = Buffer.byteLength(req.body);\n    const fileObject = { file, fileSize };\n    try {\n      // run beforeSaveFile trigger\n      const triggerResult = await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeSave,\n        fileObject,\n        config,\n        req.auth\n      );\n      let saveResult;\n      // if a new ParseFile is returned check if it's an already saved file\n      if (triggerResult instanceof Parse.File) {\n        fileObject.file = triggerResult;\n        if (triggerResult.url()) {\n          // set fileSize to null because we wont know how big it is here\n          fileObject.fileSize = null;\n          saveResult = {\n            url: triggerResult.url(),\n            name: triggerResult._name,\n          };\n        }\n      }\n      // if the file returned by the trigger has already been saved skip saving anything\n      if (!saveResult) {\n        // if the ParseFile returned is type uri, download the file before saving it\n        await addFileDataIfNeeded(fileObject.file);\n        // update fileSize\n        const bufferData = Buffer.from(fileObject.file._data, 'base64');\n        fileObject.fileSize = Buffer.byteLength(bufferData);\n        // prepare file options\n        const fileOptions = {\n          metadata: fileObject.file._metadata,\n        };\n        // some s3-compatible providers (DigitalOcean, Linode) do not accept tags\n        // so we do not include the tags option if it is empty.\n        const fileTags =\n          Object.keys(fileObject.file._tags).length > 0 ? { tags: fileObject.file._tags } : {};\n        Object.assign(fileOptions, fileTags);\n        // save file\n        const createFileResult = await filesController.createFile(\n          config,\n          fileObject.file._name,\n          bufferData,\n          fileObject.file._source.type,\n          fileOptions\n        );\n        // update file with new data\n        fileObject.file._name = createFileResult.name;\n        fileObject.file._url = createFileResult.url;\n        fileObject.file._requestTask = null;\n        fileObject.file._previousSave = Promise.resolve(fileObject.file);\n        saveResult = {\n          url: createFileResult.url,\n          name: createFileResult.name,\n        };\n      }\n      // run afterSaveFile trigger\n      await triggers.maybeRunFileTrigger(triggers.Types.afterSave, fileObject, config, req.auth);\n      res.status(201);\n      res.set('Location', saveResult.url);\n      res.json(saveResult);\n    } catch (e) {\n      logger.error('Error creating a file: ', e);\n      const error = triggers.resolveError(e, {\n        code: Parse.Error.FILE_SAVE_ERROR,\n        message: `Could not store file: ${fileObject.file._name}.`,\n      });\n      next(error);\n    }\n  }\n\n  async deleteHandler(req, res, next) {\n    try {\n      const { filesController } = req.config;\n      const { filename } = req.params;\n      // run beforeDeleteFile trigger\n      const file = new Parse.File(filename);\n      file._url = filesController.adapter.getFileLocation(req.config, filename);\n      const fileObject = { file, fileSize: null };\n      await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeDelete,\n        fileObject,\n        req.config,\n        req.auth\n      );\n      // delete file\n      await filesController.deleteFile(req.config, filename);\n      // run afterDeleteFile trigger\n      await triggers.maybeRunFileTrigger(\n        triggers.Types.afterDelete,\n        fileObject,\n        req.config,\n        req.auth\n      );\n      res.status(200);\n      // TODO: return useful JSON here?\n      res.end();\n    } catch (e) {\n      logger.error('Error deleting a file: ', e);\n      const error = triggers.resolveError(e, {\n        code: Parse.Error.FILE_DELETE_ERROR,\n        message: 'Could not delete file.',\n      });\n      next(error);\n    }\n  }\n\n  async metadataHandler(req, res) {\n    try {\n      const config = Config.get(req.params.appId);\n      const { filesController } = config;\n      const { filename } = req.params;\n      const data = await filesController.getMetadata(filename);\n      res.status(200);\n      res.json(data);\n    } catch (e) {\n      res.status(200);\n      res.json({});\n    }\n  }\n}\n\nfunction isFileStreamable(req, filesController) {\n  const range = (req.get('Range') || '/-/').split('-');\n  const start = Number(range[0]);\n  const end = Number(range[1]);\n  return (\n    (!isNaN(start) || !isNaN(end)) && typeof filesController.adapter.handleFileStream === 'function'\n  );\n}\n"]}
|
|
314
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Routers/FilesRouter.js"],"names":["triggers","require","http","Utils","downloadFileFromURI","uri","Promise","res","rej","get","response","setDefaultEncoding","body","headers","on","data","e","message","addFileDataIfNeeded","file","_source","format","base64","_previousSave","_data","_requestTask","FilesRouter","expressRouter","maxUploadSize","router","express","Router","getHandler","metadataHandler","post","req","next","Parse","Error","INVALID_FILE_NAME","BodyParser","raw","type","limit","Middlewares","handleParseHeaders","createHandler","delete","enforceMasterKeyAccess","deleteHandler","config","Config","params","appId","status","err","OPERATION_FORBIDDEN","json","code","error","filesController","filename","contentType","mime","getType","isFileStreamable","handleFileStream","catch","set","end","getFileData","then","length","user","auth","isMaster","isLinked","AnonymousUtils","fileUpload","enableForAnonymousUser","FILE_SAVE_ERROR","enableForAuthenticatedUser","enableForPublic","validateFilename","toString","File","metadata","tags","fileData","requestKeywordDenylist","keyword","match","objectContainsKeyValue","key","value","INVALID_KEY_NAME","JSON","stringify","setTags","setMetadata","fileSize","Buffer","byteLength","fileObject","triggerResult","maybeRunFileTrigger","Types","beforeSave","saveResult","url","name","_name","bufferData","from","fileOptions","_metadata","fileTags","Object","keys","_tags","assign","createFileResult","createFile","_url","resolve","afterSave","logger","resolveError","adapter","getFileLocation","beforeDelete","deleteFile","afterDelete","FILE_DELETE_ERROR","getMetadata","range","split","start","Number","isNaN"],"mappings":"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","sourcesContent":["import express from 'express';\nimport BodyParser from 'body-parser';\nimport * as Middlewares from '../middlewares';\nimport Parse from 'parse/node';\nimport Config from '../Config';\nimport mime from 'mime';\nimport logger from '../logger';\nconst triggers = require('../triggers');\nconst http = require('http');\nconst Utils = require('../Utils');\n\nconst downloadFileFromURI = uri => {\n  return new Promise((res, rej) => {\n    http\n      .get(uri, response => {\n        response.setDefaultEncoding('base64');\n        let body = `data:${response.headers['content-type']};base64,`;\n        response.on('data', data => (body += data));\n        response.on('end', () => res(body));\n      })\n      .on('error', e => {\n        rej(`Error downloading file from ${uri}: ${e.message}`);\n      });\n  });\n};\n\nconst addFileDataIfNeeded = async file => {\n  if (file._source.format === 'uri') {\n    const base64 = await downloadFileFromURI(file._source.uri);\n    file._previousSave = file;\n    file._data = base64;\n    file._requestTask = null;\n  }\n  return file;\n};\n\nexport class FilesRouter {\n  expressRouter({ maxUploadSize = '20Mb' } = {}) {\n    var router = express.Router();\n    router.get('/files/:appId/:filename', this.getHandler);\n    router.get('/files/:appId/metadata/:filename', this.metadataHandler);\n\n    router.post('/files', function (req, res, next) {\n      next(new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Filename not provided.'));\n    });\n\n    router.post(\n      '/files/:filename',\n      BodyParser.raw({\n        type: () => {\n          return true;\n        },\n        limit: maxUploadSize,\n      }), // Allow uploads without Content-Type, or with any Content-Type.\n      Middlewares.handleParseHeaders,\n      this.createHandler\n    );\n\n    router.delete(\n      '/files/:filename',\n      Middlewares.handleParseHeaders,\n      Middlewares.enforceMasterKeyAccess,\n      this.deleteHandler\n    );\n    return router;\n  }\n\n  getHandler(req, res) {\n    const config = Config.get(req.params.appId);\n    if (!config) {\n      res.status(403);\n      const err = new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, 'Invalid application ID.');\n      res.json({ code: err.code, error: err.message });\n      return;\n    }\n    const filesController = config.filesController;\n    const filename = req.params.filename;\n    const contentType = mime.getType(filename);\n    if (isFileStreamable(req, filesController)) {\n      filesController.handleFileStream(config, filename, req, res, contentType).catch(() => {\n        res.status(404);\n        res.set('Content-Type', 'text/plain');\n        res.end('File not found.');\n      });\n    } else {\n      filesController\n        .getFileData(config, filename)\n        .then(data => {\n          res.status(200);\n          res.set('Content-Type', contentType);\n          res.set('Content-Length', data.length);\n          res.end(data);\n        })\n        .catch(() => {\n          res.status(404);\n          res.set('Content-Type', 'text/plain');\n          res.end('File not found.');\n        });\n    }\n  }\n\n  async createHandler(req, res, next) {\n    const config = req.config;\n    const user = req.auth.user;\n    const isMaster = req.auth.isMaster;\n    const isLinked = user && Parse.AnonymousUtils.isLinked(user);\n    if (!isMaster && !config.fileUpload.enableForAnonymousUser && isLinked) {\n      next(\n        new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'File upload by anonymous user is disabled.')\n      );\n      return;\n    }\n    if (!isMaster && !config.fileUpload.enableForAuthenticatedUser && !isLinked && user) {\n      next(\n        new Parse.Error(\n          Parse.Error.FILE_SAVE_ERROR,\n          'File upload by authenticated user is disabled.'\n        )\n      );\n      return;\n    }\n    if (!isMaster && !config.fileUpload.enableForPublic && !user) {\n      next(new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'File upload by public is disabled.'));\n      return;\n    }\n    const filesController = config.filesController;\n    const { filename } = req.params;\n    const contentType = req.get('Content-type');\n\n    if (!req.body || !req.body.length) {\n      next(new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'Invalid file upload.'));\n      return;\n    }\n\n    const error = filesController.validateFilename(filename);\n    if (error) {\n      next(error);\n      return;\n    }\n\n    const base64 = req.body.toString('base64');\n    const file = new Parse.File(filename, { base64 }, contentType);\n    const { metadata = {}, tags = {} } = req.fileData || {};\n    if (req.config && req.config.requestKeywordDenylist) {\n      // Scan request data for denied keywords\n      for (const keyword of req.config.requestKeywordDenylist) {\n        const match =\n          Utils.objectContainsKeyValue(metadata, keyword.key, keyword.value) ||\n          Utils.objectContainsKeyValue(tags, keyword.key, keyword.value);\n        if (match) {\n          next(\n            new Parse.Error(\n              Parse.Error.INVALID_KEY_NAME,\n              `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`\n            )\n          );\n          return;\n        }\n      }\n    }\n    file.setTags(tags);\n    file.setMetadata(metadata);\n    const fileSize = Buffer.byteLength(req.body);\n    const fileObject = { file, fileSize };\n    try {\n      // run beforeSaveFile trigger\n      const triggerResult = await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeSave,\n        fileObject,\n        config,\n        req.auth\n      );\n      let saveResult;\n      // if a new ParseFile is returned check if it's an already saved file\n      if (triggerResult instanceof Parse.File) {\n        fileObject.file = triggerResult;\n        if (triggerResult.url()) {\n          // set fileSize to null because we wont know how big it is here\n          fileObject.fileSize = null;\n          saveResult = {\n            url: triggerResult.url(),\n            name: triggerResult._name,\n          };\n        }\n      }\n      // if the file returned by the trigger has already been saved skip saving anything\n      if (!saveResult) {\n        // if the ParseFile returned is type uri, download the file before saving it\n        await addFileDataIfNeeded(fileObject.file);\n        // update fileSize\n        const bufferData = Buffer.from(fileObject.file._data, 'base64');\n        fileObject.fileSize = Buffer.byteLength(bufferData);\n        // prepare file options\n        const fileOptions = {\n          metadata: fileObject.file._metadata,\n        };\n        // some s3-compatible providers (DigitalOcean, Linode) do not accept tags\n        // so we do not include the tags option if it is empty.\n        const fileTags =\n          Object.keys(fileObject.file._tags).length > 0 ? { tags: fileObject.file._tags } : {};\n        Object.assign(fileOptions, fileTags);\n        // save file\n        const createFileResult = await filesController.createFile(\n          config,\n          fileObject.file._name,\n          bufferData,\n          fileObject.file._source.type,\n          fileOptions\n        );\n        // update file with new data\n        fileObject.file._name = createFileResult.name;\n        fileObject.file._url = createFileResult.url;\n        fileObject.file._requestTask = null;\n        fileObject.file._previousSave = Promise.resolve(fileObject.file);\n        saveResult = {\n          url: createFileResult.url,\n          name: createFileResult.name,\n        };\n      }\n      // run afterSaveFile trigger\n      await triggers.maybeRunFileTrigger(triggers.Types.afterSave, fileObject, config, req.auth);\n      res.status(201);\n      res.set('Location', saveResult.url);\n      res.json(saveResult);\n    } catch (e) {\n      logger.error('Error creating a file: ', e);\n      const error = triggers.resolveError(e, {\n        code: Parse.Error.FILE_SAVE_ERROR,\n        message: `Could not store file: ${fileObject.file._name}.`,\n      });\n      next(error);\n    }\n  }\n\n  async deleteHandler(req, res, next) {\n    try {\n      const { filesController } = req.config;\n      const { filename } = req.params;\n      // run beforeDeleteFile trigger\n      const file = new Parse.File(filename);\n      file._url = filesController.adapter.getFileLocation(req.config, filename);\n      const fileObject = { file, fileSize: null };\n      await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeDelete,\n        fileObject,\n        req.config,\n        req.auth\n      );\n      // delete file\n      await filesController.deleteFile(req.config, filename);\n      // run afterDeleteFile trigger\n      await triggers.maybeRunFileTrigger(\n        triggers.Types.afterDelete,\n        fileObject,\n        req.config,\n        req.auth\n      );\n      res.status(200);\n      // TODO: return useful JSON here?\n      res.end();\n    } catch (e) {\n      logger.error('Error deleting a file: ', e);\n      const error = triggers.resolveError(e, {\n        code: Parse.Error.FILE_DELETE_ERROR,\n        message: 'Could not delete file.',\n      });\n      next(error);\n    }\n  }\n\n  async metadataHandler(req, res) {\n    try {\n      const config = Config.get(req.params.appId);\n      const { filesController } = config;\n      const { filename } = req.params;\n      const data = await filesController.getMetadata(filename);\n      res.status(200);\n      res.json(data);\n    } catch (e) {\n      res.status(200);\n      res.json({});\n    }\n  }\n}\n\nfunction isFileStreamable(req, filesController) {\n  const range = (req.get('Range') || '/-/').split('-');\n  const start = Number(range[0]);\n  const end = Number(range[1]);\n  return (\n    (!isNaN(start) || !isNaN(end)) && typeof filesController.adapter.handleFileStream === 'function'\n  );\n}\n"]}
|
|
@@ -25,6 +25,8 @@ var _middlewares = require("../middlewares");
|
|
|
25
25
|
|
|
26
26
|
var _RestWrite = _interopRequireDefault(require("../RestWrite"));
|
|
27
27
|
|
|
28
|
+
var _logger = require("../logger");
|
|
29
|
+
|
|
28
30
|
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
|
|
29
31
|
|
|
30
32
|
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) { symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); } keys.push.apply(keys, symbols); } return keys; }
|
|
@@ -205,7 +207,22 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
205
207
|
}
|
|
206
208
|
|
|
207
209
|
async handleLogIn(req) {
|
|
208
|
-
const user = await this._authenticateUserFromRequest(req);
|
|
210
|
+
const user = await this._authenticateUserFromRequest(req);
|
|
211
|
+
const authData = req.body && req.body.authData; // Check if user has provided their required auth providers
|
|
212
|
+
|
|
213
|
+
_Auth.default.checkIfUserHasProvidedConfiguredProvidersForLogin(authData, user.authData, req.config);
|
|
214
|
+
|
|
215
|
+
let authDataResponse;
|
|
216
|
+
let validatedAuthData;
|
|
217
|
+
|
|
218
|
+
if (authData) {
|
|
219
|
+
const res = await _Auth.default.handleAuthDataValidation(authData, new _RestWrite.default(req.config, req.auth, '_User', {
|
|
220
|
+
objectId: user.objectId
|
|
221
|
+
}, req.body, user, req.info.clientSDK, req.info.context), user);
|
|
222
|
+
authDataResponse = res.authDataResponse;
|
|
223
|
+
validatedAuthData = res.authData;
|
|
224
|
+
} // handle password expiry policy
|
|
225
|
+
|
|
209
226
|
|
|
210
227
|
if (req.config.passwordPolicy && req.config.passwordPolicy.maxPasswordAge) {
|
|
211
228
|
let changedAt = user._password_changed_at;
|
|
@@ -238,7 +255,15 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
238
255
|
|
|
239
256
|
await (0, _triggers.maybeRunTrigger)(_triggers.Types.beforeLogin, req.auth, _node.default.User.fromJSON(Object.assign({
|
|
240
257
|
className: '_User'
|
|
241
|
-
}, user)), null, req.config);
|
|
258
|
+
}, user)), null, req.config); // If we have some new validated authData update directly
|
|
259
|
+
|
|
260
|
+
if (validatedAuthData && Object.keys(validatedAuthData).length) {
|
|
261
|
+
await req.config.database.update('_User', {
|
|
262
|
+
objectId: user.objectId
|
|
263
|
+
}, {
|
|
264
|
+
authData: validatedAuthData
|
|
265
|
+
}, {});
|
|
266
|
+
}
|
|
242
267
|
|
|
243
268
|
const {
|
|
244
269
|
sessionData,
|
|
@@ -262,6 +287,11 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
262
287
|
(0, _triggers.maybeRunTrigger)(_triggers.Types.afterLogin, _objectSpread(_objectSpread({}, req.auth), {}, {
|
|
263
288
|
user: afterLoginUser
|
|
264
289
|
}), afterLoginUser, null, req.config);
|
|
290
|
+
|
|
291
|
+
if (authDataResponse) {
|
|
292
|
+
user.authDataResponse = authDataResponse;
|
|
293
|
+
}
|
|
294
|
+
|
|
265
295
|
return {
|
|
266
296
|
response: user
|
|
267
297
|
};
|
|
@@ -458,6 +488,137 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
458
488
|
});
|
|
459
489
|
}
|
|
460
490
|
|
|
491
|
+
async handleChallenge(req) {
|
|
492
|
+
const {
|
|
493
|
+
username,
|
|
494
|
+
email,
|
|
495
|
+
password,
|
|
496
|
+
authData,
|
|
497
|
+
challengeData
|
|
498
|
+
} = req.body; // if username or email provided with password try to authenticate the user by username
|
|
499
|
+
|
|
500
|
+
let user;
|
|
501
|
+
|
|
502
|
+
if (username || email) {
|
|
503
|
+
if (!password) {
|
|
504
|
+
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, 'You provided username or email, you need to also provide password.');
|
|
505
|
+
}
|
|
506
|
+
|
|
507
|
+
user = await this._authenticateUserFromRequest(req);
|
|
508
|
+
}
|
|
509
|
+
|
|
510
|
+
if (!challengeData) {
|
|
511
|
+
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, 'Nothing to challenge.');
|
|
512
|
+
}
|
|
513
|
+
|
|
514
|
+
if (typeof challengeData !== 'object') {
|
|
515
|
+
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, 'challengeData should be an object.');
|
|
516
|
+
}
|
|
517
|
+
|
|
518
|
+
let request;
|
|
519
|
+
let parseUser; // Try to find user by authData
|
|
520
|
+
|
|
521
|
+
if (authData) {
|
|
522
|
+
if (typeof authData !== 'object') {
|
|
523
|
+
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, 'authData should be an object.');
|
|
524
|
+
}
|
|
525
|
+
|
|
526
|
+
if (user) {
|
|
527
|
+
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, 'You cannot provide username/email and authData, only use one identification method.');
|
|
528
|
+
}
|
|
529
|
+
|
|
530
|
+
if (Object.keys(authData).filter(key => authData[key].id).length > 1) {
|
|
531
|
+
throw new _node.default.Error(_node.default.Error.OTHER_CAUSE, 'You cannot provide more than one authData provider with an id.');
|
|
532
|
+
}
|
|
533
|
+
|
|
534
|
+
const results = await _Auth.default.findUsersWithAuthData(req.config, authData);
|
|
535
|
+
|
|
536
|
+
try {
|
|
537
|
+
if (!results[0] || results.length > 1) {
|
|
538
|
+
throw new _node.default.Error(_node.default.Error.OBJECT_NOT_FOUND, 'User not found.');
|
|
539
|
+
} // Find the provider used to find the user
|
|
540
|
+
|
|
541
|
+
|
|
542
|
+
const provider = Object.keys(authData).find(key => authData[key].id);
|
|
543
|
+
parseUser = _node.default.User.fromJSON(_objectSpread({
|
|
544
|
+
className: '_User'
|
|
545
|
+
}, results[0]));
|
|
546
|
+
request = (0, _triggers.getRequestObject)(undefined, req.auth, parseUser, parseUser, req.config);
|
|
547
|
+
request.isChallenge = true; // Validate authData used to identify the user to avoid brute-force attack on `id`
|
|
548
|
+
|
|
549
|
+
const {
|
|
550
|
+
validator
|
|
551
|
+
} = req.config.authDataManager.getValidatorForProvider(provider);
|
|
552
|
+
const validatorResponse = await validator(authData[provider], req, parseUser, request);
|
|
553
|
+
|
|
554
|
+
if (validatorResponse && validatorResponse.validator) {
|
|
555
|
+
await validatorResponse.validator();
|
|
556
|
+
}
|
|
557
|
+
} catch (e) {
|
|
558
|
+
// Rewrite the error to avoid guess id attack
|
|
559
|
+
_logger.logger.error(e);
|
|
560
|
+
|
|
561
|
+
throw new _node.default.Error(_node.default.Error.OBJECT_NOT_FOUND, 'User not found.');
|
|
562
|
+
}
|
|
563
|
+
}
|
|
564
|
+
|
|
565
|
+
if (!parseUser) {
|
|
566
|
+
parseUser = user ? _node.default.User.fromJSON(_objectSpread({
|
|
567
|
+
className: '_User'
|
|
568
|
+
}, user)) : undefined;
|
|
569
|
+
}
|
|
570
|
+
|
|
571
|
+
if (!request) {
|
|
572
|
+
request = (0, _triggers.getRequestObject)(undefined, req.auth, parseUser, parseUser, req.config);
|
|
573
|
+
request.isChallenge = true;
|
|
574
|
+
}
|
|
575
|
+
|
|
576
|
+
const acc = {}; // Execute challenge step-by-step with consistent order for better error feedback
|
|
577
|
+
// and to avoid to trigger others challenges if one of them fails
|
|
578
|
+
|
|
579
|
+
for (const provider of Object.keys(challengeData).sort()) {
|
|
580
|
+
try {
|
|
581
|
+
const authAdapter = req.config.authDataManager.getValidatorForProvider(provider);
|
|
582
|
+
|
|
583
|
+
if (!authAdapter) {
|
|
584
|
+
continue;
|
|
585
|
+
}
|
|
586
|
+
|
|
587
|
+
const {
|
|
588
|
+
adapter: {
|
|
589
|
+
challenge
|
|
590
|
+
}
|
|
591
|
+
} = authAdapter;
|
|
592
|
+
|
|
593
|
+
if (typeof challenge === 'function') {
|
|
594
|
+
const providerChallengeResponse = await challenge(challengeData[provider], authData && authData[provider], req.config.auth[provider], request);
|
|
595
|
+
acc[provider] = providerChallengeResponse || true;
|
|
596
|
+
}
|
|
597
|
+
} catch (err) {
|
|
598
|
+
const e = (0, _triggers.resolveError)(err, {
|
|
599
|
+
code: _node.default.Error.SCRIPT_FAILED,
|
|
600
|
+
message: 'Challenge failed. Unknown error.'
|
|
601
|
+
});
|
|
602
|
+
const userString = req.auth && req.auth.user ? req.auth.user.id : undefined;
|
|
603
|
+
|
|
604
|
+
_logger.logger.error(`Failed running auth step challenge for ${provider} for user ${userString} with Error: ` + JSON.stringify(e), {
|
|
605
|
+
authenticationStep: 'challenge',
|
|
606
|
+
error: e,
|
|
607
|
+
user: userString,
|
|
608
|
+
provider
|
|
609
|
+
});
|
|
610
|
+
|
|
611
|
+
throw e;
|
|
612
|
+
}
|
|
613
|
+
}
|
|
614
|
+
|
|
615
|
+
return {
|
|
616
|
+
response: {
|
|
617
|
+
challengeData: acc
|
|
618
|
+
}
|
|
619
|
+
};
|
|
620
|
+
}
|
|
621
|
+
|
|
461
622
|
mountRoutes() {
|
|
462
623
|
this.route('GET', '/users', req => {
|
|
463
624
|
return this.handleFind(req);
|
|
@@ -498,6 +659,9 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
498
659
|
this.route('GET', '/verifyPassword', req => {
|
|
499
660
|
return this.handleVerifyPassword(req);
|
|
500
661
|
});
|
|
662
|
+
this.route('POST', '/challenge', req => {
|
|
663
|
+
return this.handleChallenge(req);
|
|
664
|
+
});
|
|
501
665
|
}
|
|
502
666
|
|
|
503
667
|
}
|
|
@@ -505,4 +669,4 @@ class UsersRouter extends _ClassesRouter.default {
|
|
|
505
669
|
exports.UsersRouter = UsersRouter;
|
|
506
670
|
var _default = UsersRouter;
|
|
507
671
|
exports.default = _default;
|
|
508
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Routers/UsersRouter.js"],"names":["UsersRouter","ClassesRouter","className","removeHiddenProperties","obj","key","Object","prototype","hasOwnProperty","call","test","_sanitizeAuthData","user","password","authData","keys","forEach","provider","length","_authenticateUserFromRequest","req","Promise","resolve","reject","payload","body","username","query","email","Parse","Error","USERNAME_MISSING","PASSWORD_MISSING","OBJECT_NOT_FOUND","isValidPassword","$or","config","database","find","then","results","loggerController","warn","filter","passwordCrypto","compare","correct","accountLockoutPolicy","AccountLockout","handleLoginAttempt","auth","isMaster","ACL","verifyUserEmails","preventLoginWithUnverifiedEmail","emailVerified","EMAIL_NOT_FOUND","catch","error","handleMe","info","sessionToken","INVALID_SESSION_TOKEN","rest","Auth","master","include","clientSDK","context","response","handleLogIn","passwordPolicy","maxPasswordAge","changedAt","_password_changed_at","Date","update","_encode","__type","iso","expiresAt","getTime","filesController","expandFilesInObject","TriggerTypes","beforeLogin","User","fromJSON","assign","sessionData","createSession","RestWrite","userId","objectId","createdWith","action","authProvider","installationId","afterLoginUser","afterLogin","handleLogInAs","OPERATION_FORBIDDEN","INVALID_VALUE","queryResults","handleVerifyPassword","handleLogOut","success","undefined","records","del","_runAfterLogoutTrigger","session","afterLogout","Session","_throwOnBadEmailConfig","Config","validateEmailConfiguration","emailAdapter","userController","adapter","appName","publicServerURL","emailVerifyTokenValidityDuration","emailVerifyTokenReuseIfValid","e","INTERNAL_SERVER_ERROR","handleResetRequest","EMAIL_MISSING","INVALID_EMAIL_ADDRESS","sendPasswordResetEmail","err","code","handleVerificationEmailRequest","OTHER_CAUSE","regenerateEmailVerifyToken","sendVerificationEmail","mountRoutes","route","handleFind","promiseEnsureIdempotency","handleCreate","handleGet","handleUpdate","handleDelete"],"mappings":"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","sourcesContent":["// These methods handle the User-related routes.\n\nimport Parse from 'parse/node';\nimport Config from '../Config';\nimport AccountLockout from '../AccountLockout';\nimport ClassesRouter from './ClassesRouter';\nimport rest from '../rest';\nimport Auth from '../Auth';\nimport passwordCrypto from '../password';\nimport { maybeRunTrigger, Types as TriggerTypes } from '../triggers';\nimport { promiseEnsureIdempotency } from '../middlewares';\nimport RestWrite from '../RestWrite';\n\nexport class UsersRouter extends ClassesRouter {\n  className() {\n    return '_User';\n  }\n\n  /**\n   * Removes all \"_\" prefixed properties from an object, except \"__type\"\n   * @param {Object} obj An object.\n   */\n  static removeHiddenProperties(obj) {\n    for (var key in obj) {\n      if (Object.prototype.hasOwnProperty.call(obj, key)) {\n        // Regexp comes from Parse.Object.prototype.validate\n        if (key !== '__type' && !/^[A-Za-z][0-9A-Za-z_]*$/.test(key)) {\n          delete obj[key];\n        }\n      }\n    }\n  }\n\n  /**\n   * After retrieving a user directly from the database, we need to remove the\n   * password from the object (for security), and fix an issue some SDKs have\n   * with null values\n   */\n  _sanitizeAuthData(user) {\n    delete user.password;\n\n    // Sometimes the authData still has null on that keys\n    // https://github.com/parse-community/parse-server/issues/935\n    if (user.authData) {\n      Object.keys(user.authData).forEach(provider => {\n        if (user.authData[provider] === null) {\n          delete user.authData[provider];\n        }\n      });\n      if (Object.keys(user.authData).length == 0) {\n        delete user.authData;\n      }\n    }\n  }\n\n  /**\n   * Validates a password request in login and verifyPassword\n   * @param {Object} req The request\n   * @returns {Object} User object\n   * @private\n   */\n  _authenticateUserFromRequest(req) {\n    return new Promise((resolve, reject) => {\n      // Use query parameters instead if provided in url\n      let payload = req.body;\n      if (\n        (!payload.username && req.query && req.query.username) ||\n        (!payload.email && req.query && req.query.email)\n      ) {\n        payload = req.query;\n      }\n      const { username, email, password } = payload;\n\n      // TODO: use the right error codes / descriptions.\n      if (!username && !email) {\n        throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'username/email is required.');\n      }\n      if (!password) {\n        throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'password is required.');\n      }\n      if (\n        typeof password !== 'string' ||\n        (email && typeof email !== 'string') ||\n        (username && typeof username !== 'string')\n      ) {\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n      }\n\n      let user;\n      let isValidPassword = false;\n      let query;\n      if (email && username) {\n        query = { email, username };\n      } else if (email) {\n        query = { email };\n      } else {\n        query = { $or: [{ username }, { email: username }] };\n      }\n      return req.config.database\n        .find('_User', query)\n        .then(results => {\n          if (!results.length) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n          }\n\n          if (results.length > 1) {\n            // corner case where user1 has username == user2 email\n            req.config.loggerController.warn(\n              \"There is a user which email is the same as another user's username, logging in based on username\"\n            );\n            user = results.filter(user => user.username === username)[0];\n          } else {\n            user = results[0];\n          }\n\n          return passwordCrypto.compare(password, user.password);\n        })\n        .then(correct => {\n          isValidPassword = correct;\n          const accountLockoutPolicy = new AccountLockout(user, req.config);\n          return accountLockoutPolicy.handleLoginAttempt(isValidPassword);\n        })\n        .then(() => {\n          if (!isValidPassword) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n          }\n          // Ensure the user isn't locked out\n          // A locked out user won't be able to login\n          // To lock a user out, just set the ACL to `masterKey` only  ({}).\n          // Empty ACL is OK\n          if (!req.auth.isMaster && user.ACL && Object.keys(user.ACL).length == 0) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n          }\n          if (\n            req.config.verifyUserEmails &&\n            req.config.preventLoginWithUnverifiedEmail &&\n            !user.emailVerified\n          ) {\n            throw new Parse.Error(Parse.Error.EMAIL_NOT_FOUND, 'User email is not verified.');\n          }\n\n          this._sanitizeAuthData(user);\n\n          return resolve(user);\n        })\n        .catch(error => {\n          return reject(error);\n        });\n    });\n  }\n\n  handleMe(req) {\n    if (!req.info || !req.info.sessionToken) {\n      throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n    }\n    const sessionToken = req.info.sessionToken;\n    return rest\n      .find(\n        req.config,\n        Auth.master(req.config),\n        '_Session',\n        { sessionToken },\n        { include: 'user' },\n        req.info.clientSDK,\n        req.info.context\n      )\n      .then(response => {\n        if (!response.results || response.results.length == 0 || !response.results[0].user) {\n          throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n        } else {\n          const user = response.results[0].user;\n          // Send token back on the login, because SDKs expect that.\n          user.sessionToken = sessionToken;\n\n          // Remove hidden properties.\n          UsersRouter.removeHiddenProperties(user);\n\n          return { response: user };\n        }\n      });\n  }\n\n  async handleLogIn(req) {\n    const user = await this._authenticateUserFromRequest(req);\n\n    // handle password expiry policy\n    if (req.config.passwordPolicy && req.config.passwordPolicy.maxPasswordAge) {\n      let changedAt = user._password_changed_at;\n\n      if (!changedAt) {\n        // password was created before expiry policy was enabled.\n        // simply update _User object so that it will start enforcing from now\n        changedAt = new Date();\n        req.config.database.update(\n          '_User',\n          { username: user.username },\n          { _password_changed_at: Parse._encode(changedAt) }\n        );\n      } else {\n        // check whether the password has expired\n        if (changedAt.__type == 'Date') {\n          changedAt = new Date(changedAt.iso);\n        }\n        // Calculate the expiry time.\n        const expiresAt = new Date(\n          changedAt.getTime() + 86400000 * req.config.passwordPolicy.maxPasswordAge\n        );\n        if (expiresAt < new Date())\n          // fail of current time is past password expiry time\n          throw new Parse.Error(\n            Parse.Error.OBJECT_NOT_FOUND,\n            'Your password has expired. Please reset your password.'\n          );\n      }\n    }\n\n    // Remove hidden properties.\n    UsersRouter.removeHiddenProperties(user);\n\n    req.config.filesController.expandFilesInObject(req.config, user);\n\n    // Before login trigger; throws if failure\n    await maybeRunTrigger(\n      TriggerTypes.beforeLogin,\n      req.auth,\n      Parse.User.fromJSON(Object.assign({ className: '_User' }, user)),\n      null,\n      req.config\n    );\n\n    const { sessionData, createSession } = RestWrite.createSession(req.config, {\n      userId: user.objectId,\n      createdWith: {\n        action: 'login',\n        authProvider: 'password',\n      },\n      installationId: req.info.installationId,\n    });\n\n    user.sessionToken = sessionData.sessionToken;\n\n    await createSession();\n\n    const afterLoginUser = Parse.User.fromJSON(Object.assign({ className: '_User' }, user));\n    maybeRunTrigger(\n      TriggerTypes.afterLogin,\n      { ...req.auth, user: afterLoginUser },\n      afterLoginUser,\n      null,\n      req.config\n    );\n\n    return { response: user };\n  }\n\n  /**\n   * This allows master-key clients to create user sessions without access to\n   * user credentials. This enables systems that can authenticate access another\n   * way (API key, app administrators) to act on a user's behalf.\n   *\n   * We create a new session rather than looking for an existing session; we\n   * want this to work in situations where the user is logged out on all\n   * devices, since this can be used by automated systems acting on the user's\n   * behalf.\n   *\n   * For the moment, we're omitting event hooks and lockout checks, since\n   * immediate use cases suggest /loginAs could be used for semantically\n   * different reasons from /login\n   */\n  async handleLogInAs(req) {\n    if (!req.auth.isMaster) {\n      throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, 'master key is required');\n    }\n\n    const userId = req.body.userId || req.query.userId;\n    if (!userId) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_VALUE,\n        'userId must not be empty, null, or undefined'\n      );\n    }\n\n    const queryResults = await req.config.database.find('_User', { objectId: userId });\n    const user = queryResults[0];\n    if (!user) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'user not found');\n    }\n\n    this._sanitizeAuthData(user);\n\n    const { sessionData, createSession } = RestWrite.createSession(req.config, {\n      userId,\n      createdWith: {\n        action: 'login',\n        authProvider: 'masterkey',\n      },\n      installationId: req.info.installationId,\n    });\n\n    user.sessionToken = sessionData.sessionToken;\n\n    await createSession();\n\n    return { response: user };\n  }\n\n  handleVerifyPassword(req) {\n    return this._authenticateUserFromRequest(req)\n      .then(user => {\n        // Remove hidden properties.\n        UsersRouter.removeHiddenProperties(user);\n\n        return { response: user };\n      })\n      .catch(error => {\n        throw error;\n      });\n  }\n\n  handleLogOut(req) {\n    const success = { response: {} };\n    if (req.info && req.info.sessionToken) {\n      return rest\n        .find(\n          req.config,\n          Auth.master(req.config),\n          '_Session',\n          { sessionToken: req.info.sessionToken },\n          undefined,\n          req.info.clientSDK,\n          req.info.context\n        )\n        .then(records => {\n          if (records.results && records.results.length) {\n            return rest\n              .del(\n                req.config,\n                Auth.master(req.config),\n                '_Session',\n                records.results[0].objectId,\n                req.info.context\n              )\n              .then(() => {\n                this._runAfterLogoutTrigger(req, records.results[0]);\n                return Promise.resolve(success);\n              });\n          }\n          return Promise.resolve(success);\n        });\n    }\n    return Promise.resolve(success);\n  }\n\n  _runAfterLogoutTrigger(req, session) {\n    // After logout trigger\n    maybeRunTrigger(\n      TriggerTypes.afterLogout,\n      req.auth,\n      Parse.Session.fromJSON(Object.assign({ className: '_Session' }, session)),\n      null,\n      req.config\n    );\n  }\n\n  _throwOnBadEmailConfig(req) {\n    try {\n      Config.validateEmailConfiguration({\n        emailAdapter: req.config.userController.adapter,\n        appName: req.config.appName,\n        publicServerURL: req.config.publicServerURL,\n        emailVerifyTokenValidityDuration: req.config.emailVerifyTokenValidityDuration,\n        emailVerifyTokenReuseIfValid: req.config.emailVerifyTokenReuseIfValid,\n      });\n    } catch (e) {\n      if (typeof e === 'string') {\n        // Maybe we need a Bad Configuration error, but the SDKs won't understand it. For now, Internal Server Error.\n        throw new Parse.Error(\n          Parse.Error.INTERNAL_SERVER_ERROR,\n          'An appName, publicServerURL, and emailAdapter are required for password reset and email verification functionality.'\n        );\n      } else {\n        throw e;\n      }\n    }\n  }\n\n  handleResetRequest(req) {\n    this._throwOnBadEmailConfig(req);\n\n    const { email } = req.body;\n    if (!email) {\n      throw new Parse.Error(Parse.Error.EMAIL_MISSING, 'you must provide an email');\n    }\n    if (typeof email !== 'string') {\n      throw new Parse.Error(\n        Parse.Error.INVALID_EMAIL_ADDRESS,\n        'you must provide a valid email string'\n      );\n    }\n    const userController = req.config.userController;\n    return userController.sendPasswordResetEmail(email).then(\n      () => {\n        return Promise.resolve({\n          response: {},\n        });\n      },\n      err => {\n        if (err.code === Parse.Error.OBJECT_NOT_FOUND) {\n          // Return success so that this endpoint can't\n          // be used to enumerate valid emails\n          return Promise.resolve({\n            response: {},\n          });\n        } else {\n          throw err;\n        }\n      }\n    );\n  }\n\n  handleVerificationEmailRequest(req) {\n    this._throwOnBadEmailConfig(req);\n\n    const { email } = req.body;\n    if (!email) {\n      throw new Parse.Error(Parse.Error.EMAIL_MISSING, 'you must provide an email');\n    }\n    if (typeof email !== 'string') {\n      throw new Parse.Error(\n        Parse.Error.INVALID_EMAIL_ADDRESS,\n        'you must provide a valid email string'\n      );\n    }\n\n    return req.config.database.find('_User', { email: email }).then(results => {\n      if (!results.length || results.length < 1) {\n        throw new Parse.Error(Parse.Error.EMAIL_NOT_FOUND, `No user found with email ${email}`);\n      }\n      const user = results[0];\n\n      // remove password field, messes with saving on postgres\n      delete user.password;\n\n      if (user.emailVerified) {\n        throw new Parse.Error(Parse.Error.OTHER_CAUSE, `Email ${email} is already verified.`);\n      }\n\n      const userController = req.config.userController;\n      return userController.regenerateEmailVerifyToken(user).then(() => {\n        userController.sendVerificationEmail(user);\n        return { response: {} };\n      });\n    });\n  }\n\n  mountRoutes() {\n    this.route('GET', '/users', req => {\n      return this.handleFind(req);\n    });\n    this.route('POST', '/users', promiseEnsureIdempotency, req => {\n      return this.handleCreate(req);\n    });\n    this.route('GET', '/users/me', req => {\n      return this.handleMe(req);\n    });\n    this.route('GET', '/users/:objectId', req => {\n      return this.handleGet(req);\n    });\n    this.route('PUT', '/users/:objectId', promiseEnsureIdempotency, req => {\n      return this.handleUpdate(req);\n    });\n    this.route('DELETE', '/users/:objectId', req => {\n      return this.handleDelete(req);\n    });\n    this.route('GET', '/login', req => {\n      return this.handleLogIn(req);\n    });\n    this.route('POST', '/login', req => {\n      return this.handleLogIn(req);\n    });\n    this.route('POST', '/loginAs', req => {\n      return this.handleLogInAs(req);\n    });\n    this.route('POST', '/logout', req => {\n      return this.handleLogOut(req);\n    });\n    this.route('POST', '/requestPasswordReset', req => {\n      return this.handleResetRequest(req);\n    });\n    this.route('POST', '/verificationEmailRequest', req => {\n      return this.handleVerificationEmailRequest(req);\n    });\n    this.route('GET', '/verifyPassword', req => {\n      return this.handleVerifyPassword(req);\n    });\n  }\n}\n\nexport default UsersRouter;\n"]}
|
|
672
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Routers/UsersRouter.js"],"names":["UsersRouter","ClassesRouter","className","removeHiddenProperties","obj","key","Object","prototype","hasOwnProperty","call","test","_sanitizeAuthData","user","password","authData","keys","forEach","provider","length","_authenticateUserFromRequest","req","Promise","resolve","reject","payload","body","username","query","email","Parse","Error","USERNAME_MISSING","PASSWORD_MISSING","OBJECT_NOT_FOUND","isValidPassword","$or","config","database","find","then","results","loggerController","warn","filter","passwordCrypto","compare","correct","accountLockoutPolicy","AccountLockout","handleLoginAttempt","auth","isMaster","ACL","verifyUserEmails","preventLoginWithUnverifiedEmail","emailVerified","EMAIL_NOT_FOUND","catch","error","handleMe","info","sessionToken","INVALID_SESSION_TOKEN","rest","Auth","master","include","clientSDK","context","response","handleLogIn","checkIfUserHasProvidedConfiguredProvidersForLogin","authDataResponse","validatedAuthData","res","handleAuthDataValidation","RestWrite","objectId","passwordPolicy","maxPasswordAge","changedAt","_password_changed_at","Date","update","_encode","__type","iso","expiresAt","getTime","filesController","expandFilesInObject","TriggerTypes","beforeLogin","User","fromJSON","assign","sessionData","createSession","userId","createdWith","action","authProvider","installationId","afterLoginUser","afterLogin","handleLogInAs","OPERATION_FORBIDDEN","INVALID_VALUE","queryResults","handleVerifyPassword","handleLogOut","success","undefined","records","del","_runAfterLogoutTrigger","session","afterLogout","Session","_throwOnBadEmailConfig","Config","validateEmailConfiguration","emailAdapter","userController","adapter","appName","publicServerURL","emailVerifyTokenValidityDuration","emailVerifyTokenReuseIfValid","e","INTERNAL_SERVER_ERROR","handleResetRequest","EMAIL_MISSING","INVALID_EMAIL_ADDRESS","sendPasswordResetEmail","err","code","handleVerificationEmailRequest","OTHER_CAUSE","regenerateEmailVerifyToken","sendVerificationEmail","handleChallenge","challengeData","request","parseUser","id","findUsersWithAuthData","isChallenge","validator","authDataManager","getValidatorForProvider","validatorResponse","logger","acc","sort","authAdapter","challenge","providerChallengeResponse","SCRIPT_FAILED","message","userString","JSON","stringify","authenticationStep","mountRoutes","route","handleFind","promiseEnsureIdempotency","handleCreate","handleGet","handleUpdate","handleDelete"],"mappings":"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","sourcesContent":["// These methods handle the User-related routes.\n\nimport Parse from 'parse/node';\nimport Config from '../Config';\nimport AccountLockout from '../AccountLockout';\nimport ClassesRouter from './ClassesRouter';\nimport rest from '../rest';\nimport Auth from '../Auth';\nimport passwordCrypto from '../password';\nimport {\n  maybeRunTrigger,\n  Types as TriggerTypes,\n  getRequestObject,\n  resolveError,\n} from '../triggers';\nimport { promiseEnsureIdempotency } from '../middlewares';\nimport RestWrite from '../RestWrite';\nimport { logger } from '../logger';\n\nexport class UsersRouter extends ClassesRouter {\n  className() {\n    return '_User';\n  }\n\n  /**\n   * Removes all \"_\" prefixed properties from an object, except \"__type\"\n   * @param {Object} obj An object.\n   */\n  static removeHiddenProperties(obj) {\n    for (var key in obj) {\n      if (Object.prototype.hasOwnProperty.call(obj, key)) {\n        // Regexp comes from Parse.Object.prototype.validate\n        if (key !== '__type' && !/^[A-Za-z][0-9A-Za-z_]*$/.test(key)) {\n          delete obj[key];\n        }\n      }\n    }\n  }\n\n  /**\n   * After retrieving a user directly from the database, we need to remove the\n   * password from the object (for security), and fix an issue some SDKs have\n   * with null values\n   */\n  _sanitizeAuthData(user) {\n    delete user.password;\n\n    // Sometimes the authData still has null on that keys\n    // https://github.com/parse-community/parse-server/issues/935\n    if (user.authData) {\n      Object.keys(user.authData).forEach(provider => {\n        if (user.authData[provider] === null) {\n          delete user.authData[provider];\n        }\n      });\n      if (Object.keys(user.authData).length == 0) {\n        delete user.authData;\n      }\n    }\n  }\n\n  /**\n   * Validates a password request in login and verifyPassword\n   * @param {Object} req The request\n   * @returns {Object} User object\n   * @private\n   */\n  _authenticateUserFromRequest(req) {\n    return new Promise((resolve, reject) => {\n      // Use query parameters instead if provided in url\n      let payload = req.body;\n      if (\n        (!payload.username && req.query && req.query.username) ||\n        (!payload.email && req.query && req.query.email)\n      ) {\n        payload = req.query;\n      }\n      const { username, email, password } = payload;\n\n      // TODO: use the right error codes / descriptions.\n      if (!username && !email) {\n        throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'username/email is required.');\n      }\n      if (!password) {\n        throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'password is required.');\n      }\n      if (\n        typeof password !== 'string' ||\n        (email && typeof email !== 'string') ||\n        (username && typeof username !== 'string')\n      ) {\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n      }\n\n      let user;\n      let isValidPassword = false;\n      let query;\n      if (email && username) {\n        query = { email, username };\n      } else if (email) {\n        query = { email };\n      } else {\n        query = { $or: [{ username }, { email: username }] };\n      }\n      return req.config.database\n        .find('_User', query)\n        .then(results => {\n          if (!results.length) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n          }\n\n          if (results.length > 1) {\n            // corner case where user1 has username == user2 email\n            req.config.loggerController.warn(\n              \"There is a user which email is the same as another user's username, logging in based on username\"\n            );\n            user = results.filter(user => user.username === username)[0];\n          } else {\n            user = results[0];\n          }\n\n          return passwordCrypto.compare(password, user.password);\n        })\n        .then(correct => {\n          isValidPassword = correct;\n          const accountLockoutPolicy = new AccountLockout(user, req.config);\n          return accountLockoutPolicy.handleLoginAttempt(isValidPassword);\n        })\n        .then(() => {\n          if (!isValidPassword) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n          }\n          // Ensure the user isn't locked out\n          // A locked out user won't be able to login\n          // To lock a user out, just set the ACL to `masterKey` only  ({}).\n          // Empty ACL is OK\n          if (!req.auth.isMaster && user.ACL && Object.keys(user.ACL).length == 0) {\n            throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'Invalid username/password.');\n          }\n          if (\n            req.config.verifyUserEmails &&\n            req.config.preventLoginWithUnverifiedEmail &&\n            !user.emailVerified\n          ) {\n            throw new Parse.Error(Parse.Error.EMAIL_NOT_FOUND, 'User email is not verified.');\n          }\n\n          this._sanitizeAuthData(user);\n\n          return resolve(user);\n        })\n        .catch(error => {\n          return reject(error);\n        });\n    });\n  }\n\n  handleMe(req) {\n    if (!req.info || !req.info.sessionToken) {\n      throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n    }\n    const sessionToken = req.info.sessionToken;\n    return rest\n      .find(\n        req.config,\n        Auth.master(req.config),\n        '_Session',\n        { sessionToken },\n        { include: 'user' },\n        req.info.clientSDK,\n        req.info.context\n      )\n      .then(response => {\n        if (!response.results || response.results.length == 0 || !response.results[0].user) {\n          throw new Parse.Error(Parse.Error.INVALID_SESSION_TOKEN, 'Invalid session token');\n        } else {\n          const user = response.results[0].user;\n          // Send token back on the login, because SDKs expect that.\n          user.sessionToken = sessionToken;\n\n          // Remove hidden properties.\n          UsersRouter.removeHiddenProperties(user);\n          return { response: user };\n        }\n      });\n  }\n\n  async handleLogIn(req) {\n    const user = await this._authenticateUserFromRequest(req);\n    const authData = req.body && req.body.authData;\n    // Check if user has provided their required auth providers\n    Auth.checkIfUserHasProvidedConfiguredProvidersForLogin(authData, user.authData, req.config);\n\n    let authDataResponse;\n    let validatedAuthData;\n    if (authData) {\n      const res = await Auth.handleAuthDataValidation(\n        authData,\n        new RestWrite(\n          req.config,\n          req.auth,\n          '_User',\n          { objectId: user.objectId },\n          req.body,\n          user,\n          req.info.clientSDK,\n          req.info.context\n        ),\n        user\n      );\n      authDataResponse = res.authDataResponse;\n      validatedAuthData = res.authData;\n    }\n\n    // handle password expiry policy\n    if (req.config.passwordPolicy && req.config.passwordPolicy.maxPasswordAge) {\n      let changedAt = user._password_changed_at;\n\n      if (!changedAt) {\n        // password was created before expiry policy was enabled.\n        // simply update _User object so that it will start enforcing from now\n        changedAt = new Date();\n        req.config.database.update(\n          '_User',\n          { username: user.username },\n          { _password_changed_at: Parse._encode(changedAt) }\n        );\n      } else {\n        // check whether the password has expired\n        if (changedAt.__type == 'Date') {\n          changedAt = new Date(changedAt.iso);\n        }\n        // Calculate the expiry time.\n        const expiresAt = new Date(\n          changedAt.getTime() + 86400000 * req.config.passwordPolicy.maxPasswordAge\n        );\n        if (expiresAt < new Date())\n          // fail of current time is past password expiry time\n          throw new Parse.Error(\n            Parse.Error.OBJECT_NOT_FOUND,\n            'Your password has expired. Please reset your password.'\n          );\n      }\n    }\n\n    // Remove hidden properties.\n    UsersRouter.removeHiddenProperties(user);\n\n    req.config.filesController.expandFilesInObject(req.config, user);\n\n    // Before login trigger; throws if failure\n    await maybeRunTrigger(\n      TriggerTypes.beforeLogin,\n      req.auth,\n      Parse.User.fromJSON(Object.assign({ className: '_User' }, user)),\n      null,\n      req.config\n    );\n\n    // If we have some new validated authData update directly\n    if (validatedAuthData && Object.keys(validatedAuthData).length) {\n      await req.config.database.update(\n        '_User',\n        { objectId: user.objectId },\n        { authData: validatedAuthData },\n        {}\n      );\n    }\n\n    const { sessionData, createSession } = RestWrite.createSession(req.config, {\n      userId: user.objectId,\n      createdWith: {\n        action: 'login',\n        authProvider: 'password',\n      },\n      installationId: req.info.installationId,\n    });\n\n    user.sessionToken = sessionData.sessionToken;\n\n    await createSession();\n\n    const afterLoginUser = Parse.User.fromJSON(Object.assign({ className: '_User' }, user));\n    maybeRunTrigger(\n      TriggerTypes.afterLogin,\n      { ...req.auth, user: afterLoginUser },\n      afterLoginUser,\n      null,\n      req.config\n    );\n\n    if (authDataResponse) {\n      user.authDataResponse = authDataResponse;\n    }\n\n    return { response: user };\n  }\n\n  /**\n   * This allows master-key clients to create user sessions without access to\n   * user credentials. This enables systems that can authenticate access another\n   * way (API key, app administrators) to act on a user's behalf.\n   *\n   * We create a new session rather than looking for an existing session; we\n   * want this to work in situations where the user is logged out on all\n   * devices, since this can be used by automated systems acting on the user's\n   * behalf.\n   *\n   * For the moment, we're omitting event hooks and lockout checks, since\n   * immediate use cases suggest /loginAs could be used for semantically\n   * different reasons from /login\n   */\n  async handleLogInAs(req) {\n    if (!req.auth.isMaster) {\n      throw new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, 'master key is required');\n    }\n\n    const userId = req.body.userId || req.query.userId;\n    if (!userId) {\n      throw new Parse.Error(\n        Parse.Error.INVALID_VALUE,\n        'userId must not be empty, null, or undefined'\n      );\n    }\n\n    const queryResults = await req.config.database.find('_User', { objectId: userId });\n    const user = queryResults[0];\n    if (!user) {\n      throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'user not found');\n    }\n\n    this._sanitizeAuthData(user);\n\n    const { sessionData, createSession } = RestWrite.createSession(req.config, {\n      userId,\n      createdWith: {\n        action: 'login',\n        authProvider: 'masterkey',\n      },\n      installationId: req.info.installationId,\n    });\n\n    user.sessionToken = sessionData.sessionToken;\n\n    await createSession();\n\n    return { response: user };\n  }\n\n  handleVerifyPassword(req) {\n    return this._authenticateUserFromRequest(req)\n      .then(user => {\n        // Remove hidden properties.\n        UsersRouter.removeHiddenProperties(user);\n\n        return { response: user };\n      })\n      .catch(error => {\n        throw error;\n      });\n  }\n\n  handleLogOut(req) {\n    const success = { response: {} };\n    if (req.info && req.info.sessionToken) {\n      return rest\n        .find(\n          req.config,\n          Auth.master(req.config),\n          '_Session',\n          { sessionToken: req.info.sessionToken },\n          undefined,\n          req.info.clientSDK,\n          req.info.context\n        )\n        .then(records => {\n          if (records.results && records.results.length) {\n            return rest\n              .del(\n                req.config,\n                Auth.master(req.config),\n                '_Session',\n                records.results[0].objectId,\n                req.info.context\n              )\n              .then(() => {\n                this._runAfterLogoutTrigger(req, records.results[0]);\n                return Promise.resolve(success);\n              });\n          }\n          return Promise.resolve(success);\n        });\n    }\n    return Promise.resolve(success);\n  }\n\n  _runAfterLogoutTrigger(req, session) {\n    // After logout trigger\n    maybeRunTrigger(\n      TriggerTypes.afterLogout,\n      req.auth,\n      Parse.Session.fromJSON(Object.assign({ className: '_Session' }, session)),\n      null,\n      req.config\n    );\n  }\n\n  _throwOnBadEmailConfig(req) {\n    try {\n      Config.validateEmailConfiguration({\n        emailAdapter: req.config.userController.adapter,\n        appName: req.config.appName,\n        publicServerURL: req.config.publicServerURL,\n        emailVerifyTokenValidityDuration: req.config.emailVerifyTokenValidityDuration,\n        emailVerifyTokenReuseIfValid: req.config.emailVerifyTokenReuseIfValid,\n      });\n    } catch (e) {\n      if (typeof e === 'string') {\n        // Maybe we need a Bad Configuration error, but the SDKs won't understand it. For now, Internal Server Error.\n        throw new Parse.Error(\n          Parse.Error.INTERNAL_SERVER_ERROR,\n          'An appName, publicServerURL, and emailAdapter are required for password reset and email verification functionality.'\n        );\n      } else {\n        throw e;\n      }\n    }\n  }\n\n  handleResetRequest(req) {\n    this._throwOnBadEmailConfig(req);\n\n    const { email } = req.body;\n    if (!email) {\n      throw new Parse.Error(Parse.Error.EMAIL_MISSING, 'you must provide an email');\n    }\n    if (typeof email !== 'string') {\n      throw new Parse.Error(\n        Parse.Error.INVALID_EMAIL_ADDRESS,\n        'you must provide a valid email string'\n      );\n    }\n    const userController = req.config.userController;\n    return userController.sendPasswordResetEmail(email).then(\n      () => {\n        return Promise.resolve({\n          response: {},\n        });\n      },\n      err => {\n        if (err.code === Parse.Error.OBJECT_NOT_FOUND) {\n          // Return success so that this endpoint can't\n          // be used to enumerate valid emails\n          return Promise.resolve({\n            response: {},\n          });\n        } else {\n          throw err;\n        }\n      }\n    );\n  }\n\n  handleVerificationEmailRequest(req) {\n    this._throwOnBadEmailConfig(req);\n\n    const { email } = req.body;\n    if (!email) {\n      throw new Parse.Error(Parse.Error.EMAIL_MISSING, 'you must provide an email');\n    }\n    if (typeof email !== 'string') {\n      throw new Parse.Error(\n        Parse.Error.INVALID_EMAIL_ADDRESS,\n        'you must provide a valid email string'\n      );\n    }\n\n    return req.config.database.find('_User', { email: email }).then(results => {\n      if (!results.length || results.length < 1) {\n        throw new Parse.Error(Parse.Error.EMAIL_NOT_FOUND, `No user found with email ${email}`);\n      }\n      const user = results[0];\n\n      // remove password field, messes with saving on postgres\n      delete user.password;\n\n      if (user.emailVerified) {\n        throw new Parse.Error(Parse.Error.OTHER_CAUSE, `Email ${email} is already verified.`);\n      }\n\n      const userController = req.config.userController;\n      return userController.regenerateEmailVerifyToken(user).then(() => {\n        userController.sendVerificationEmail(user);\n        return { response: {} };\n      });\n    });\n  }\n\n  async handleChallenge(req) {\n    const { username, email, password, authData, challengeData } = req.body;\n\n    // if username or email provided with password try to authenticate the user by username\n    let user;\n    if (username || email) {\n      if (!password) {\n        throw new Parse.Error(\n          Parse.Error.OTHER_CAUSE,\n          'You provided username or email, you need to also provide password.'\n        );\n      }\n      user = await this._authenticateUserFromRequest(req);\n    }\n\n    if (!challengeData) {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'Nothing to challenge.');\n    }\n\n    if (typeof challengeData !== 'object') {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'challengeData should be an object.');\n    }\n\n    let request;\n    let parseUser;\n\n    // Try to find user by authData\n    if (authData) {\n      if (typeof authData !== 'object') {\n        throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'authData should be an object.');\n      }\n      if (user) {\n        throw new Parse.Error(\n          Parse.Error.OTHER_CAUSE,\n          'You cannot provide username/email and authData, only use one identification method.'\n        );\n      }\n\n      if (Object.keys(authData).filter(key => authData[key].id).length > 1) {\n        throw new Parse.Error(\n          Parse.Error.OTHER_CAUSE,\n          'You cannot provide more than one authData provider with an id.'\n        );\n      }\n\n      const results = await Auth.findUsersWithAuthData(req.config, authData);\n\n      try {\n        if (!results[0] || results.length > 1) {\n          throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'User not found.');\n        }\n        // Find the provider used to find the user\n        const provider = Object.keys(authData).find(key => authData[key].id);\n\n        parseUser = Parse.User.fromJSON({ className: '_User', ...results[0] });\n        request = getRequestObject(undefined, req.auth, parseUser, parseUser, req.config);\n        request.isChallenge = true;\n        // Validate authData used to identify the user to avoid brute-force attack on `id`\n        const { validator } = req.config.authDataManager.getValidatorForProvider(provider);\n        const validatorResponse = await validator(authData[provider], req, parseUser, request);\n        if (validatorResponse && validatorResponse.validator) {\n          await validatorResponse.validator();\n        }\n      } catch (e) {\n        // Rewrite the error to avoid guess id attack\n        logger.error(e);\n        throw new Parse.Error(Parse.Error.OBJECT_NOT_FOUND, 'User not found.');\n      }\n    }\n\n    if (!parseUser) {\n      parseUser = user ? Parse.User.fromJSON({ className: '_User', ...user }) : undefined;\n    }\n\n    if (!request) {\n      request = getRequestObject(undefined, req.auth, parseUser, parseUser, req.config);\n      request.isChallenge = true;\n    }\n    const acc = {};\n    // Execute challenge step-by-step with consistent order for better error feedback\n    // and to avoid to trigger others challenges if one of them fails\n    for (const provider of Object.keys(challengeData).sort()) {\n      try {\n        const authAdapter = req.config.authDataManager.getValidatorForProvider(provider);\n        if (!authAdapter) {\n          continue;\n        }\n        const {\n          adapter: { challenge },\n        } = authAdapter;\n        if (typeof challenge === 'function') {\n          const providerChallengeResponse = await challenge(\n            challengeData[provider],\n            authData && authData[provider],\n            req.config.auth[provider],\n            request\n          );\n          acc[provider] = providerChallengeResponse || true;\n        }\n      } catch (err) {\n        const e = resolveError(err, {\n          code: Parse.Error.SCRIPT_FAILED,\n          message: 'Challenge failed. Unknown error.',\n        });\n        const userString = req.auth && req.auth.user ? req.auth.user.id : undefined;\n        logger.error(\n          `Failed running auth step challenge for ${provider} for user ${userString} with Error: ` +\n            JSON.stringify(e),\n          {\n            authenticationStep: 'challenge',\n            error: e,\n            user: userString,\n            provider,\n          }\n        );\n        throw e;\n      }\n    }\n    return { response: { challengeData: acc } };\n  }\n\n  mountRoutes() {\n    this.route('GET', '/users', req => {\n      return this.handleFind(req);\n    });\n    this.route('POST', '/users', promiseEnsureIdempotency, req => {\n      return this.handleCreate(req);\n    });\n    this.route('GET', '/users/me', req => {\n      return this.handleMe(req);\n    });\n    this.route('GET', '/users/:objectId', req => {\n      return this.handleGet(req);\n    });\n    this.route('PUT', '/users/:objectId', promiseEnsureIdempotency, req => {\n      return this.handleUpdate(req);\n    });\n    this.route('DELETE', '/users/:objectId', req => {\n      return this.handleDelete(req);\n    });\n    this.route('GET', '/login', req => {\n      return this.handleLogIn(req);\n    });\n    this.route('POST', '/login', req => {\n      return this.handleLogIn(req);\n    });\n    this.route('POST', '/loginAs', req => {\n      return this.handleLogInAs(req);\n    });\n    this.route('POST', '/logout', req => {\n      return this.handleLogOut(req);\n    });\n    this.route('POST', '/requestPasswordReset', req => {\n      return this.handleResetRequest(req);\n    });\n    this.route('POST', '/verificationEmailRequest', req => {\n      return this.handleVerificationEmailRequest(req);\n    });\n    this.route('GET', '/verifyPassword', req => {\n      return this.handleVerifyPassword(req);\n    });\n    this.route('POST', '/challenge', req => {\n      return this.handleChallenge(req);\n    });\n  }\n}\n\nexport default UsersRouter;\n"]}
|