parse-server 6.3.0-beta.1 → 6.3.0

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.
@@ -157,15 +157,12 @@ class FilesRouter {
157
157
  metadata = {},
158
158
  tags = {}
159
159
  } = req.fileData || {};
160
- if (req.config && req.config.requestKeywordDenylist) {
161
- // Scan request data for denied keywords
162
- for (const keyword of req.config.requestKeywordDenylist) {
163
- const match = Utils.objectContainsKeyValue(metadata, keyword.key, keyword.value) || Utils.objectContainsKeyValue(tags, keyword.key, keyword.value);
164
- if (match) {
165
- next(new _node.default.Error(_node.default.Error.INVALID_KEY_NAME, `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`));
166
- return;
167
- }
168
- }
160
+ try {
161
+ Utils.checkProhibitedKeywords(config, metadata);
162
+ Utils.checkProhibitedKeywords(config, tags);
163
+ } catch (error) {
164
+ next(new _node.default.Error(_node.default.Error.INVALID_KEY_NAME, error));
165
+ return;
169
166
  }
170
167
  file.setTags(tags);
171
168
  file.setMetadata(metadata);
@@ -290,4 +287,4 @@ function isFileStreamable(req, filesController) {
290
287
  const end = Number(range[1]);
291
288
  return (!isNaN(start) || !isNaN(end)) && typeof filesController.adapter.handleFileStream === 'function';
292
289
  }
293
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"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","handleParseSession","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","fileExtensions","isValidExtension","extension","some","ext","regex","RegExp","test","includes","split","join","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","start","Number","isNaN"],"sources":["../../src/Routers/FilesRouter.js"],"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      Middlewares.handleParseSession,\n      this.createHandler\n    );\n\n    router.delete(\n      '/files/:filename',\n      Middlewares.handleParseHeaders,\n      Middlewares.handleParseSession,\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 fileExtensions = config.fileUpload?.fileExtensions;\n    if (!isMaster && fileExtensions) {\n      const isValidExtension = extension => {\n        return fileExtensions.some(ext => {\n          if (ext === '*') {\n            return true;\n          }\n          const regex = new RegExp(fileExtensions);\n          if (regex.test(extension)) {\n            return true;\n          }\n        });\n      };\n      let extension = contentType;\n      if (filename && filename.includes('.')) {\n        extension = filename.split('.')[1];\n      } else if (contentType && contentType.includes('/')) {\n        extension = contentType.split('/')[1];\n      }\n      extension = extension.split(' ').join('');\n\n      if (!isValidExtension(extension)) {\n        next(\n          new Parse.Error(\n            Parse.Error.FILE_SAVE_ERROR,\n            `File upload of extension ${extension} is disabled.`\n          )\n        );\n        return;\n      }\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"],"mappings":";;;;;;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AAA+B;AAAA;AAAA;AAC/B,MAAMA,QAAQ,GAAGC,OAAO,CAAC,aAAa,CAAC;AACvC,MAAMC,IAAI,GAAGD,OAAO,CAAC,MAAM,CAAC;AAC5B,MAAME,KAAK,GAAGF,OAAO,CAAC,UAAU,CAAC;AAEjC,MAAMG,mBAAmB,GAAGC,GAAG,IAAI;EACjC,OAAO,IAAIC,OAAO,CAAC,CAACC,GAAG,EAAEC,GAAG,KAAK;IAC/BN,IAAI,CACDO,GAAG,CAACJ,GAAG,EAAEK,QAAQ,IAAI;MACpBA,QAAQ,CAACC,kBAAkB,CAAC,QAAQ,CAAC;MACrC,IAAIC,IAAI,GAAI,QAAOF,QAAQ,CAACG,OAAO,CAAC,cAAc,CAAE,UAAS;MAC7DH,QAAQ,CAACI,EAAE,CAAC,MAAM,EAAEC,IAAI,IAAKH,IAAI,IAAIG,IAAK,CAAC;MAC3CL,QAAQ,CAACI,EAAE,CAAC,KAAK,EAAE,MAAMP,GAAG,CAACK,IAAI,CAAC,CAAC;IACrC,CAAC,CAAC,CACDE,EAAE,CAAC,OAAO,EAAEE,CAAC,IAAI;MAChBR,GAAG,CAAE,+BAA8BH,GAAI,KAAIW,CAAC,CAACC,OAAQ,EAAC,CAAC;IACzD,CAAC,CAAC;EACN,CAAC,CAAC;AACJ,CAAC;AAED,MAAMC,mBAAmB,GAAG,MAAMC,IAAI,IAAI;EACxC,IAAIA,IAAI,CAACC,OAAO,CAACC,MAAM,KAAK,KAAK,EAAE;IACjC,MAAMC,MAAM,GAAG,MAAMlB,mBAAmB,CAACe,IAAI,CAACC,OAAO,CAACf,GAAG,CAAC;IAC1Dc,IAAI,CAACI,aAAa,GAAGJ,IAAI;IACzBA,IAAI,CAACK,KAAK,GAAGF,MAAM;IACnBH,IAAI,CAACM,YAAY,GAAG,IAAI;EAC1B;EACA,OAAON,IAAI;AACb,CAAC;AAEM,MAAMO,WAAW,CAAC;EACvBC,aAAa,CAAC;IAAEC,aAAa,GAAG;EAAO,CAAC,GAAG,CAAC,CAAC,EAAE;IAC7C,IAAIC,MAAM,GAAGC,gBAAO,CAACC,MAAM,EAAE;IAC7BF,MAAM,CAACpB,GAAG,CAAC,yBAAyB,EAAE,IAAI,CAACuB,UAAU,CAAC;IACtDH,MAAM,CAACpB,GAAG,CAAC,kCAAkC,EAAE,IAAI,CAACwB,eAAe,CAAC;IAEpEJ,MAAM,CAACK,IAAI,CAAC,QAAQ,EAAE,UAAUC,GAAG,EAAE5B,GAAG,EAAE6B,IAAI,EAAE;MAC9CA,IAAI,CAAC,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACC,iBAAiB,EAAE,wBAAwB,CAAC,CAAC;IAChF,CAAC,CAAC;IAEFV,MAAM,CAACK,IAAI,CACT,kBAAkB,EAClBM,mBAAU,CAACC,GAAG,CAAC;MACbC,IAAI,EAAE,MAAM;QACV,OAAO,IAAI;MACb,CAAC;MACDC,KAAK,EAAEf;IACT,CAAC,CAAC;IAAE;IACJgB,WAAW,CAACC,kBAAkB,EAC9BD,WAAW,CAACE,kBAAkB,EAC9B,IAAI,CAACC,aAAa,CACnB;IAEDlB,MAAM,CAACmB,MAAM,CACX,kBAAkB,EAClBJ,WAAW,CAACC,kBAAkB,EAC9BD,WAAW,CAACE,kBAAkB,EAC9BF,WAAW,CAACK,sBAAsB,EAClC,IAAI,CAACC,aAAa,CACnB;IACD,OAAOrB,MAAM;EACf;EAEAG,UAAU,CAACG,GAAG,EAAE5B,GAAG,EAAE;IACnB,MAAM4C,MAAM,GAAGC,eAAM,CAAC3C,GAAG,CAAC0B,GAAG,CAACkB,MAAM,CAACC,KAAK,CAAC;IAC3C,IAAI,CAACH,MAAM,EAAE;MACX5C,GAAG,CAACgD,MAAM,CAAC,GAAG,CAAC;MACf,MAAMC,GAAG,GAAG,IAAInB,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACmB,mBAAmB,EAAE,yBAAyB,CAAC;MACvFlD,GAAG,CAACmD,IAAI,CAAC;QAAEC,IAAI,EAAEH,GAAG,CAACG,IAAI;QAAEC,KAAK,EAAEJ,GAAG,CAACvC;MAAQ,CAAC,CAAC;MAChD;IACF;IACA,MAAM4C,eAAe,GAAGV,MAAM,CAACU,eAAe;IAC9C,MAAMC,QAAQ,GAAG3B,GAAG,CAACkB,MAAM,CAACS,QAAQ;IACpC,MAAMC,WAAW,GAAGC,aAAI,CAACC,OAAO,CAACH,QAAQ,CAAC;IAC1C,IAAII,gBAAgB,CAAC/B,GAAG,EAAE0B,eAAe,CAAC,EAAE;MAC1CA,eAAe,CAACM,gBAAgB,CAAChB,MAAM,EAAEW,QAAQ,EAAE3B,GAAG,EAAE5B,GAAG,EAAEwD,WAAW,CAAC,CAACK,KAAK,CAAC,MAAM;QACpF7D,GAAG,CAACgD,MAAM,CAAC,GAAG,CAAC;QACfhD,GAAG,CAAC8D,GAAG,CAAC,cAAc,EAAE,YAAY,CAAC;QACrC9D,GAAG,CAAC+D,GAAG,CAAC,iBAAiB,CAAC;MAC5B,CAAC,CAAC;IACJ,CAAC,MAAM;MACLT,eAAe,CACZU,WAAW,CAACpB,MAAM,EAAEW,QAAQ,CAAC,CAC7BU,IAAI,CAACzD,IAAI,IAAI;QACZR,GAAG,CAACgD,MAAM,CAAC,GAAG,CAAC;QACfhD,GAAG,CAAC8D,GAAG,CAAC,cAAc,EAAEN,WAAW,CAAC;QACpCxD,GAAG,CAAC8D,GAAG,CAAC,gBAAgB,EAAEtD,IAAI,CAAC0D,MAAM,CAAC;QACtClE,GAAG,CAAC+D,GAAG,CAACvD,IAAI,CAAC;MACf,CAAC,CAAC,CACDqD,KAAK,CAAC,MAAM;QACX7D,GAAG,CAACgD,MAAM,CAAC,GAAG,CAAC;QACfhD,GAAG,CAAC8D,GAAG,CAAC,cAAc,EAAE,YAAY,CAAC;QACrC9D,GAAG,CAAC+D,GAAG,CAAC,iBAAiB,CAAC;MAC5B,CAAC,CAAC;IACN;EACF;EAEA,MAAMvB,aAAa,CAACZ,GAAG,EAAE5B,GAAG,EAAE6B,IAAI,EAAE;IAAA;IAClC,MAAMe,MAAM,GAAGhB,GAAG,CAACgB,MAAM;IACzB,MAAMuB,IAAI,GAAGvC,GAAG,CAACwC,IAAI,CAACD,IAAI;IAC1B,MAAME,QAAQ,GAAGzC,GAAG,CAACwC,IAAI,CAACC,QAAQ;IAClC,MAAMC,QAAQ,GAAGH,IAAI,IAAIrC,aAAK,CAACyC,cAAc,CAACD,QAAQ,CAACH,IAAI,CAAC;IAC5D,IAAI,CAACE,QAAQ,IAAI,CAACzB,MAAM,CAAC4B,UAAU,CAACC,sBAAsB,IAAIH,QAAQ,EAAE;MACtEzC,IAAI,CACF,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAAC2C,eAAe,EAAE,4CAA4C,CAAC,CAC3F;MACD;IACF;IACA,IAAI,CAACL,QAAQ,IAAI,CAACzB,MAAM,CAAC4B,UAAU,CAACG,0BAA0B,IAAI,CAACL,QAAQ,IAAIH,IAAI,EAAE;MACnFtC,IAAI,CACF,IAAIC,aAAK,CAACC,KAAK,CACbD,aAAK,CAACC,KAAK,CAAC2C,eAAe,EAC3B,gDAAgD,CACjD,CACF;MACD;IACF;IACA,IAAI,CAACL,QAAQ,IAAI,CAACzB,MAAM,CAAC4B,UAAU,CAACI,eAAe,IAAI,CAACT,IAAI,EAAE;MAC5DtC,IAAI,CAAC,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAAC2C,eAAe,EAAE,oCAAoC,CAAC,CAAC;MACxF;IACF;IACA,MAAMpB,eAAe,GAAGV,MAAM,CAACU,eAAe;IAC9C,MAAM;MAAEC;IAAS,CAAC,GAAG3B,GAAG,CAACkB,MAAM;IAC/B,MAAMU,WAAW,GAAG5B,GAAG,CAAC1B,GAAG,CAAC,cAAc,CAAC;IAE3C,IAAI,CAAC0B,GAAG,CAACvB,IAAI,IAAI,CAACuB,GAAG,CAACvB,IAAI,CAAC6D,MAAM,EAAE;MACjCrC,IAAI,CAAC,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAAC2C,eAAe,EAAE,sBAAsB,CAAC,CAAC;MAC1E;IACF;IAEA,MAAMrB,KAAK,GAAGC,eAAe,CAACuB,gBAAgB,CAACtB,QAAQ,CAAC;IACxD,IAAIF,KAAK,EAAE;MACTxB,IAAI,CAACwB,KAAK,CAAC;MACX;IACF;IAEA,MAAMyB,cAAc,yBAAGlC,MAAM,CAAC4B,UAAU,uDAAjB,mBAAmBM,cAAc;IACxD,IAAI,CAACT,QAAQ,IAAIS,cAAc,EAAE;MAC/B,MAAMC,gBAAgB,GAAGC,SAAS,IAAI;QACpC,OAAOF,cAAc,CAACG,IAAI,CAACC,GAAG,IAAI;UAChC,IAAIA,GAAG,KAAK,GAAG,EAAE;YACf,OAAO,IAAI;UACb;UACA,MAAMC,KAAK,GAAG,IAAIC,MAAM,CAACN,cAAc,CAAC;UACxC,IAAIK,KAAK,CAACE,IAAI,CAACL,SAAS,CAAC,EAAE;YACzB,OAAO,IAAI;UACb;QACF,CAAC,CAAC;MACJ,CAAC;MACD,IAAIA,SAAS,GAAGxB,WAAW;MAC3B,IAAID,QAAQ,IAAIA,QAAQ,CAAC+B,QAAQ,CAAC,GAAG,CAAC,EAAE;QACtCN,SAAS,GAAGzB,QAAQ,CAACgC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;MACpC,CAAC,MAAM,IAAI/B,WAAW,IAAIA,WAAW,CAAC8B,QAAQ,CAAC,GAAG,CAAC,EAAE;QACnDN,SAAS,GAAGxB,WAAW,CAAC+B,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;MACvC;MACAP,SAAS,GAAGA,SAAS,CAACO,KAAK,CAAC,GAAG,CAAC,CAACC,IAAI,CAAC,EAAE,CAAC;MAEzC,IAAI,CAACT,gBAAgB,CAACC,SAAS,CAAC,EAAE;QAChCnD,IAAI,CACF,IAAIC,aAAK,CAACC,KAAK,CACbD,aAAK,CAACC,KAAK,CAAC2C,eAAe,EAC1B,4BAA2BM,SAAU,eAAc,CACrD,CACF;QACD;MACF;IACF;IAEA,MAAMjE,MAAM,GAAGa,GAAG,CAACvB,IAAI,CAACoF,QAAQ,CAAC,QAAQ,CAAC;IAC1C,MAAM7E,IAAI,GAAG,IAAIkB,aAAK,CAAC4D,IAAI,CAACnC,QAAQ,EAAE;MAAExC;IAAO,CAAC,EAAEyC,WAAW,CAAC;IAC9D,MAAM;MAAEmC,QAAQ,GAAG,CAAC,CAAC;MAAEC,IAAI,GAAG,CAAC;IAAE,CAAC,GAAGhE,GAAG,CAACiE,QAAQ,IAAI,CAAC,CAAC;IACvD,IAAIjE,GAAG,CAACgB,MAAM,IAAIhB,GAAG,CAACgB,MAAM,CAACkD,sBAAsB,EAAE;MACnD;MACA,KAAK,MAAMC,OAAO,IAAInE,GAAG,CAACgB,MAAM,CAACkD,sBAAsB,EAAE;QACvD,MAAME,KAAK,GACTpG,KAAK,CAACqG,sBAAsB,CAACN,QAAQ,EAAEI,OAAO,CAACG,GAAG,EAAEH,OAAO,CAACI,KAAK,CAAC,IAClEvG,KAAK,CAACqG,sBAAsB,CAACL,IAAI,EAAEG,OAAO,CAACG,GAAG,EAAEH,OAAO,CAACI,KAAK,CAAC;QAChE,IAAIH,KAAK,EAAE;UACTnE,IAAI,CACF,IAAIC,aAAK,CAACC,KAAK,CACbD,aAAK,CAACC,KAAK,CAACqE,gBAAgB,EAC3B,uCAAsCC,IAAI,CAACC,SAAS,CAACP,OAAO,CAAE,GAAE,CAClE,CACF;UACD;QACF;MACF;IACF;IACAnF,IAAI,CAAC2F,OAAO,CAACX,IAAI,CAAC;IAClBhF,IAAI,CAAC4F,WAAW,CAACb,QAAQ,CAAC;IAC1B,MAAMc,QAAQ,GAAGC,MAAM,CAACC,UAAU,CAAC/E,GAAG,CAACvB,IAAI,CAAC;IAC5C,MAAMuG,UAAU,GAAG;MAAEhG,IAAI;MAAE6F;IAAS,CAAC;IACrC,IAAI;MACF;MACA,MAAMI,aAAa,GAAG,MAAMpH,QAAQ,CAACqH,mBAAmB,CACtDrH,QAAQ,CAACsH,KAAK,CAACC,UAAU,EACzBJ,UAAU,EACVhE,MAAM,EACNhB,GAAG,CAACwC,IAAI,CACT;MACD,IAAI6C,UAAU;MACd;MACA,IAAIJ,aAAa,YAAY/E,aAAK,CAAC4D,IAAI,EAAE;QACvCkB,UAAU,CAAChG,IAAI,GAAGiG,aAAa;QAC/B,IAAIA,aAAa,CAACK,GAAG,EAAE,EAAE;UACvB;UACAN,UAAU,CAACH,QAAQ,GAAG,IAAI;UAC1BQ,UAAU,GAAG;YACXC,GAAG,EAAEL,aAAa,CAACK,GAAG,EAAE;YACxBC,IAAI,EAAEN,aAAa,CAACO;UACtB,CAAC;QACH;MACF;MACA;MACA,IAAI,CAACH,UAAU,EAAE;QACf;QACA,MAAMtG,mBAAmB,CAACiG,UAAU,CAAChG,IAAI,CAAC;QAC1C;QACA,MAAMyG,UAAU,GAAGX,MAAM,CAACY,IAAI,CAACV,UAAU,CAAChG,IAAI,CAACK,KAAK,EAAE,QAAQ,CAAC;QAC/D2F,UAAU,CAACH,QAAQ,GAAGC,MAAM,CAACC,UAAU,CAACU,UAAU,CAAC;QACnD;QACA,MAAME,WAAW,GAAG;UAClB5B,QAAQ,EAAEiB,UAAU,CAAChG,IAAI,CAAC4G;QAC5B,CAAC;QACD;QACA;QACA,MAAMC,QAAQ,GACZC,MAAM,CAACC,IAAI,CAACf,UAAU,CAAChG,IAAI,CAACgH,KAAK,CAAC,CAAC1D,MAAM,GAAG,CAAC,GAAG;UAAE0B,IAAI,EAAEgB,UAAU,CAAChG,IAAI,CAACgH;QAAM,CAAC,GAAG,CAAC,CAAC;QACtFF,MAAM,CAACG,MAAM,CAACN,WAAW,EAAEE,QAAQ,CAAC;QACpC;QACA,MAAMK,gBAAgB,GAAG,MAAMxE,eAAe,CAACyE,UAAU,CACvDnF,MAAM,EACNgE,UAAU,CAAChG,IAAI,CAACwG,KAAK,EACrBC,UAAU,EACVT,UAAU,CAAChG,IAAI,CAACC,OAAO,CAACsB,IAAI,EAC5BoF,WAAW,CACZ;QACD;QACAX,UAAU,CAAChG,IAAI,CAACwG,KAAK,GAAGU,gBAAgB,CAACX,IAAI;QAC7CP,UAAU,CAAChG,IAAI,CAACoH,IAAI,GAAGF,gBAAgB,CAACZ,GAAG;QAC3CN,UAAU,CAAChG,IAAI,CAACM,YAAY,GAAG,IAAI;QACnC0F,UAAU,CAAChG,IAAI,CAACI,aAAa,GAAGjB,OAAO,CAACkI,OAAO,CAACrB,UAAU,CAAChG,IAAI,CAAC;QAChEqG,UAAU,GAAG;UACXC,GAAG,EAAEY,gBAAgB,CAACZ,GAAG;UACzBC,IAAI,EAAEW,gBAAgB,CAACX;QACzB,CAAC;MACH;MACA;MACA,MAAM1H,QAAQ,CAACqH,mBAAmB,CAACrH,QAAQ,CAACsH,KAAK,CAACmB,SAAS,EAAEtB,UAAU,EAAEhE,MAAM,EAAEhB,GAAG,CAACwC,IAAI,CAAC;MAC1FpE,GAAG,CAACgD,MAAM,CAAC,GAAG,CAAC;MACfhD,GAAG,CAAC8D,GAAG,CAAC,UAAU,EAAEmD,UAAU,CAACC,GAAG,CAAC;MACnClH,GAAG,CAACmD,IAAI,CAAC8D,UAAU,CAAC;IACtB,CAAC,CAAC,OAAOxG,CAAC,EAAE;MACV0H,eAAM,CAAC9E,KAAK,CAAC,yBAAyB,EAAE5C,CAAC,CAAC;MAC1C,MAAM4C,KAAK,GAAG5D,QAAQ,CAAC2I,YAAY,CAAC3H,CAAC,EAAE;QACrC2C,IAAI,EAAEtB,aAAK,CAACC,KAAK,CAAC2C,eAAe;QACjChE,OAAO,EAAG,yBAAwBkG,UAAU,CAAChG,IAAI,CAACwG,KAAM;MAC1D,CAAC,CAAC;MACFvF,IAAI,CAACwB,KAAK,CAAC;IACb;EACF;EAEA,MAAMV,aAAa,CAACf,GAAG,EAAE5B,GAAG,EAAE6B,IAAI,EAAE;IAClC,IAAI;MACF,MAAM;QAAEyB;MAAgB,CAAC,GAAG1B,GAAG,CAACgB,MAAM;MACtC,MAAM;QAAEW;MAAS,CAAC,GAAG3B,GAAG,CAACkB,MAAM;MAC/B;MACA,MAAMlC,IAAI,GAAG,IAAIkB,aAAK,CAAC4D,IAAI,CAACnC,QAAQ,CAAC;MACrC3C,IAAI,CAACoH,IAAI,GAAG1E,eAAe,CAAC+E,OAAO,CAACC,eAAe,CAAC1G,GAAG,CAACgB,MAAM,EAAEW,QAAQ,CAAC;MACzE,MAAMqD,UAAU,GAAG;QAAEhG,IAAI;QAAE6F,QAAQ,EAAE;MAAK,CAAC;MAC3C,MAAMhH,QAAQ,CAACqH,mBAAmB,CAChCrH,QAAQ,CAACsH,KAAK,CAACwB,YAAY,EAC3B3B,UAAU,EACVhF,GAAG,CAACgB,MAAM,EACVhB,GAAG,CAACwC,IAAI,CACT;MACD;MACA,MAAMd,eAAe,CAACkF,UAAU,CAAC5G,GAAG,CAACgB,MAAM,EAAEW,QAAQ,CAAC;MACtD;MACA,MAAM9D,QAAQ,CAACqH,mBAAmB,CAChCrH,QAAQ,CAACsH,KAAK,CAAC0B,WAAW,EAC1B7B,UAAU,EACVhF,GAAG,CAACgB,MAAM,EACVhB,GAAG,CAACwC,IAAI,CACT;MACDpE,GAAG,CAACgD,MAAM,CAAC,GAAG,CAAC;MACf;MACAhD,GAAG,CAAC+D,GAAG,EAAE;IACX,CAAC,CAAC,OAAOtD,CAAC,EAAE;MACV0H,eAAM,CAAC9E,KAAK,CAAC,yBAAyB,EAAE5C,CAAC,CAAC;MAC1C,MAAM4C,KAAK,GAAG5D,QAAQ,CAAC2I,YAAY,CAAC3H,CAAC,EAAE;QACrC2C,IAAI,EAAEtB,aAAK,CAACC,KAAK,CAAC2G,iBAAiB;QACnChI,OAAO,EAAE;MACX,CAAC,CAAC;MACFmB,IAAI,CAACwB,KAAK,CAAC;IACb;EACF;EAEA,MAAM3B,eAAe,CAACE,GAAG,EAAE5B,GAAG,EAAE;IAC9B,IAAI;MACF,MAAM4C,MAAM,GAAGC,eAAM,CAAC3C,GAAG,CAAC0B,GAAG,CAACkB,MAAM,CAACC,KAAK,CAAC;MAC3C,MAAM;QAAEO;MAAgB,CAAC,GAAGV,MAAM;MAClC,MAAM;QAAEW;MAAS,CAAC,GAAG3B,GAAG,CAACkB,MAAM;MAC/B,MAAMtC,IAAI,GAAG,MAAM8C,eAAe,CAACqF,WAAW,CAACpF,QAAQ,CAAC;MACxDvD,GAAG,CAACgD,MAAM,CAAC,GAAG,CAAC;MACfhD,GAAG,CAACmD,IAAI,CAAC3C,IAAI,CAAC;IAChB,CAAC,CAAC,OAAOC,CAAC,EAAE;MACVT,GAAG,CAACgD,MAAM,CAAC,GAAG,CAAC;MACfhD,GAAG,CAACmD,IAAI,CAAC,CAAC,CAAC,CAAC;IACd;EACF;AACF;AAAC;AAED,SAASQ,gBAAgB,CAAC/B,GAAG,EAAE0B,eAAe,EAAE;EAC9C,MAAMsF,KAAK,GAAG,CAAChH,GAAG,CAAC1B,GAAG,CAAC,OAAO,CAAC,IAAI,KAAK,EAAEqF,KAAK,CAAC,GAAG,CAAC;EACpD,MAAMsD,KAAK,GAAGC,MAAM,CAACF,KAAK,CAAC,CAAC,CAAC,CAAC;EAC9B,MAAM7E,GAAG,GAAG+E,MAAM,CAACF,KAAK,CAAC,CAAC,CAAC,CAAC;EAC5B,OACE,CAAC,CAACG,KAAK,CAACF,KAAK,CAAC,IAAI,CAACE,KAAK,CAAChF,GAAG,CAAC,KAAK,OAAOT,eAAe,CAAC+E,OAAO,CAACzE,gBAAgB,KAAK,UAAU;AAEpG"}
290
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"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","handleParseSession","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","fileExtensions","isValidExtension","extension","some","ext","regex","RegExp","test","includes","split","join","toString","File","metadata","tags","fileData","checkProhibitedKeywords","INVALID_KEY_NAME","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","start","Number","isNaN"],"sources":["../../src/Routers/FilesRouter.js"],"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      Middlewares.handleParseSession,\n      this.createHandler\n    );\n\n    router.delete(\n      '/files/:filename',\n      Middlewares.handleParseHeaders,\n      Middlewares.handleParseSession,\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 fileExtensions = config.fileUpload?.fileExtensions;\n    if (!isMaster && fileExtensions) {\n      const isValidExtension = extension => {\n        return fileExtensions.some(ext => {\n          if (ext === '*') {\n            return true;\n          }\n          const regex = new RegExp(fileExtensions);\n          if (regex.test(extension)) {\n            return true;\n          }\n        });\n      };\n      let extension = contentType;\n      if (filename && filename.includes('.')) {\n        extension = filename.split('.')[1];\n      } else if (contentType && contentType.includes('/')) {\n        extension = contentType.split('/')[1];\n      }\n      extension = extension.split(' ').join('');\n\n      if (!isValidExtension(extension)) {\n        next(\n          new Parse.Error(\n            Parse.Error.FILE_SAVE_ERROR,\n            `File upload of extension ${extension} is disabled.`\n          )\n        );\n        return;\n      }\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    try {\n      Utils.checkProhibitedKeywords(config, metadata);\n      Utils.checkProhibitedKeywords(config, tags);\n    } catch (error) {\n      next(new Parse.Error(Parse.Error.INVALID_KEY_NAME, error));\n      return;\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"],"mappings":"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"}
@@ -14,7 +14,7 @@ function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { de
14
14
 
15
15
  var Parse = require('parse/node').Parse,
16
16
  triggers = require('../triggers');
17
- function parseObject(obj) {
17
+ function parseObject(obj, config) {
18
18
  if (Array.isArray(obj)) {
19
19
  return obj.map(item => {
20
20
  return parseObject(item);
@@ -23,20 +23,20 @@ function parseObject(obj) {
23
23
  return Object.assign(new Date(obj.iso), obj);
24
24
  } else if (obj && obj.__type == 'File') {
25
25
  return Parse.File.fromJSON(obj);
26
- } else if (obj && obj.__type == 'Pointer') {
26
+ } else if (obj && obj.__type == 'Pointer' && config.encodeParseObjectInCloudFunction) {
27
27
  return Parse.Object.fromJSON({
28
28
  __type: 'Pointer',
29
29
  className: obj.className,
30
30
  objectId: obj.objectId
31
31
  });
32
32
  } else if (obj && typeof obj === 'object') {
33
- return parseParams(obj);
33
+ return parseParams(obj, config);
34
34
  } else {
35
35
  return obj;
36
36
  }
37
37
  }
38
- function parseParams(params) {
39
- return _lodash.default.mapValues(params, parseObject);
38
+ function parseParams(params, config) {
39
+ return _lodash.default.mapValues(params, item => parseObject(item, config));
40
40
  }
41
41
  class FunctionsRouter extends _PromiseRouter.default {
42
42
  mountRoutes() {
@@ -57,7 +57,7 @@ class FunctionsRouter extends _PromiseRouter.default {
57
57
  throw new Parse.Error(Parse.Error.SCRIPT_FAILED, 'Invalid job.');
58
58
  }
59
59
  let params = Object.assign({}, req.body, req.query);
60
- params = parseParams(params);
60
+ params = parseParams(params, req.config);
61
61
  const request = {
62
62
  params: params,
63
63
  log: req.config.loggerController,
@@ -109,7 +109,7 @@ class FunctionsRouter extends _PromiseRouter.default {
109
109
  throw new Parse.Error(Parse.Error.SCRIPT_FAILED, `Invalid function: "${functionName}"`);
110
110
  }
111
111
  let params = Object.assign({}, req.body, req.query);
112
- params = parseParams(params);
112
+ params = parseParams(params, req.config);
113
113
  const request = {
114
114
  params: params,
115
115
  master: req.auth && req.auth.isMaster,
@@ -161,4 +161,4 @@ class FunctionsRouter extends _PromiseRouter.default {
161
161
  }
162
162
  }
163
163
  exports.FunctionsRouter = FunctionsRouter;
164
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","triggers","parseObject","obj","Array","isArray","map","item","__type","Object","assign","Date","iso","File","fromJSON","className","objectId","parseParams","params","_","mapValues","FunctionsRouter","PromiseRouter","mountRoutes","route","promiseEnsureIdempotency","handleCloudFunction","promiseEnforceMasterKeyAccess","req","handleCloudJob","jobName","body","applicationId","config","jobHandler","jobStatusHandler","jobFunction","getJob","Error","SCRIPT_FAILED","query","request","log","loggerController","headers","ip","message","setMessage","bind","setRunning","then","jobStatus","jobId","process","nextTick","Promise","resolve","result","setSucceeded","error","setFailed","response","createResponseObject","reject","success","_encode","resolveError","functionName","theFunction","getFunction","master","auth","isMaster","user","installationId","info","context","userString","id","undefined","cleanInput","logger","truncateLogMessage","JSON","stringify","cleanResult","logLevels","cloudFunctionSuccess","e","cloudFunctionError","maybeRunValidator"],"sources":["../../src/Routers/FunctionsRouter.js"],"sourcesContent":["// FunctionsRouter.js\n\nvar Parse = require('parse/node').Parse,\n  triggers = require('../triggers');\n\nimport PromiseRouter from '../PromiseRouter';\nimport { promiseEnforceMasterKeyAccess, promiseEnsureIdempotency } from '../middlewares';\nimport { jobStatusHandler } from '../StatusHandler';\nimport _ from 'lodash';\nimport { logger } from '../logger';\n\nfunction parseObject(obj) {\n  if (Array.isArray(obj)) {\n    return obj.map(item => {\n      return parseObject(item);\n    });\n  } else if (obj && obj.__type == 'Date') {\n    return Object.assign(new Date(obj.iso), obj);\n  } else if (obj && obj.__type == 'File') {\n    return Parse.File.fromJSON(obj);\n  } else if (obj && obj.__type == 'Pointer') {\n    return Parse.Object.fromJSON({\n      __type: 'Pointer',\n      className: obj.className,\n      objectId: obj.objectId,\n    });\n  } else if (obj && typeof obj === 'object') {\n    return parseParams(obj);\n  } else {\n    return obj;\n  }\n}\n\nfunction parseParams(params) {\n  return _.mapValues(params, parseObject);\n}\n\nexport class FunctionsRouter extends PromiseRouter {\n  mountRoutes() {\n    this.route(\n      'POST',\n      '/functions/:functionName',\n      promiseEnsureIdempotency,\n      FunctionsRouter.handleCloudFunction\n    );\n    this.route(\n      'POST',\n      '/jobs/:jobName',\n      promiseEnsureIdempotency,\n      promiseEnforceMasterKeyAccess,\n      function (req) {\n        return FunctionsRouter.handleCloudJob(req);\n      }\n    );\n    this.route('POST', '/jobs', promiseEnforceMasterKeyAccess, function (req) {\n      return FunctionsRouter.handleCloudJob(req);\n    });\n  }\n\n  static handleCloudJob(req) {\n    const jobName = req.params.jobName || req.body.jobName;\n    const applicationId = req.config.applicationId;\n    const jobHandler = jobStatusHandler(req.config);\n    const jobFunction = triggers.getJob(jobName, applicationId);\n    if (!jobFunction) {\n      throw new Parse.Error(Parse.Error.SCRIPT_FAILED, 'Invalid job.');\n    }\n    let params = Object.assign({}, req.body, req.query);\n    params = parseParams(params);\n    const request = {\n      params: params,\n      log: req.config.loggerController,\n      headers: req.config.headers,\n      ip: req.config.ip,\n      jobName,\n      message: jobHandler.setMessage.bind(jobHandler),\n    };\n\n    return jobHandler.setRunning(jobName, params).then(jobStatus => {\n      request.jobId = jobStatus.objectId;\n      // run the function async\n      process.nextTick(() => {\n        Promise.resolve()\n          .then(() => {\n            return jobFunction(request);\n          })\n          .then(\n            result => {\n              jobHandler.setSucceeded(result);\n            },\n            error => {\n              jobHandler.setFailed(error);\n            }\n          );\n      });\n      return {\n        headers: {\n          'X-Parse-Job-Status-Id': jobStatus.objectId,\n        },\n        response: {},\n      };\n    });\n  }\n\n  static createResponseObject(resolve, reject) {\n    return {\n      success: function (result) {\n        resolve({\n          response: {\n            result: Parse._encode(result),\n          },\n        });\n      },\n      error: function (message) {\n        const error = triggers.resolveError(message);\n        reject(error);\n      },\n    };\n  }\n  static handleCloudFunction(req) {\n    const functionName = req.params.functionName;\n    const applicationId = req.config.applicationId;\n    const theFunction = triggers.getFunction(functionName, applicationId);\n\n    if (!theFunction) {\n      throw new Parse.Error(Parse.Error.SCRIPT_FAILED, `Invalid function: \"${functionName}\"`);\n    }\n    let params = Object.assign({}, req.body, req.query);\n    params = parseParams(params);\n    const request = {\n      params: params,\n      master: req.auth && req.auth.isMaster,\n      user: req.auth && req.auth.user,\n      installationId: req.info.installationId,\n      log: req.config.loggerController,\n      headers: req.config.headers,\n      ip: req.config.ip,\n      functionName,\n      context: req.info.context,\n    };\n\n    return new Promise(function (resolve, reject) {\n      const userString = req.auth && req.auth.user ? req.auth.user.id : undefined;\n      const cleanInput = logger.truncateLogMessage(JSON.stringify(params));\n      const { success, error } = FunctionsRouter.createResponseObject(\n        result => {\n          try {\n            const cleanResult = logger.truncateLogMessage(JSON.stringify(result.response.result));\n            logger[req.config.logLevels.cloudFunctionSuccess](\n              `Ran cloud function ${functionName} for user ${userString} with:\\n  Input: ${cleanInput}\\n  Result: ${cleanResult}`,\n              {\n                functionName,\n                params,\n                user: userString,\n              }\n            );\n            resolve(result);\n          } catch (e) {\n            reject(e);\n          }\n        },\n        error => {\n          try {\n            logger[req.config.logLevels.cloudFunctionError](\n              `Failed running cloud function ${functionName} for user ${userString} with:\\n  Input: ${cleanInput}\\n  Error: ` +\n                JSON.stringify(error),\n              {\n                functionName,\n                error,\n                params,\n                user: userString,\n              }\n            );\n            reject(error);\n          } catch (e) {\n            reject(e);\n          }\n        }\n      );\n      return Promise.resolve()\n        .then(() => {\n          return triggers.maybeRunValidator(request, functionName, req.auth);\n        })\n        .then(() => {\n          return theFunction(request);\n        })\n        .then(success, error);\n    });\n  }\n}\n"],"mappings":"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"}
164
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","triggers","parseObject","obj","config","Array","isArray","map","item","__type","Object","assign","Date","iso","File","fromJSON","encodeParseObjectInCloudFunction","className","objectId","parseParams","params","_","mapValues","FunctionsRouter","PromiseRouter","mountRoutes","route","promiseEnsureIdempotency","handleCloudFunction","promiseEnforceMasterKeyAccess","req","handleCloudJob","jobName","body","applicationId","jobHandler","jobStatusHandler","jobFunction","getJob","Error","SCRIPT_FAILED","query","request","log","loggerController","headers","ip","message","setMessage","bind","setRunning","then","jobStatus","jobId","process","nextTick","Promise","resolve","result","setSucceeded","error","setFailed","response","createResponseObject","reject","success","_encode","resolveError","functionName","theFunction","getFunction","master","auth","isMaster","user","installationId","info","context","userString","id","undefined","cleanInput","logger","truncateLogMessage","JSON","stringify","cleanResult","logLevels","cloudFunctionSuccess","e","cloudFunctionError","maybeRunValidator"],"sources":["../../src/Routers/FunctionsRouter.js"],"sourcesContent":["// FunctionsRouter.js\n\nvar Parse = require('parse/node').Parse,\n  triggers = require('../triggers');\n\nimport PromiseRouter from '../PromiseRouter';\nimport { promiseEnforceMasterKeyAccess, promiseEnsureIdempotency } from '../middlewares';\nimport { jobStatusHandler } from '../StatusHandler';\nimport _ from 'lodash';\nimport { logger } from '../logger';\n\nfunction parseObject(obj, config) {\n  if (Array.isArray(obj)) {\n    return obj.map(item => {\n      return parseObject(item);\n    });\n  } else if (obj && obj.__type == 'Date') {\n    return Object.assign(new Date(obj.iso), obj);\n  } else if (obj && obj.__type == 'File') {\n    return Parse.File.fromJSON(obj);\n  } else if (obj && obj.__type == 'Pointer' && config.encodeParseObjectInCloudFunction) {\n    return Parse.Object.fromJSON({\n      __type: 'Pointer',\n      className: obj.className,\n      objectId: obj.objectId,\n    });\n  } else if (obj && typeof obj === 'object') {\n    return parseParams(obj, config);\n  } else {\n    return obj;\n  }\n}\n\nfunction parseParams(params, config) {\n  return _.mapValues(params, item => parseObject(item, config));\n}\n\nexport class FunctionsRouter extends PromiseRouter {\n  mountRoutes() {\n    this.route(\n      'POST',\n      '/functions/:functionName',\n      promiseEnsureIdempotency,\n      FunctionsRouter.handleCloudFunction\n    );\n    this.route(\n      'POST',\n      '/jobs/:jobName',\n      promiseEnsureIdempotency,\n      promiseEnforceMasterKeyAccess,\n      function (req) {\n        return FunctionsRouter.handleCloudJob(req);\n      }\n    );\n    this.route('POST', '/jobs', promiseEnforceMasterKeyAccess, function (req) {\n      return FunctionsRouter.handleCloudJob(req);\n    });\n  }\n\n  static handleCloudJob(req) {\n    const jobName = req.params.jobName || req.body.jobName;\n    const applicationId = req.config.applicationId;\n    const jobHandler = jobStatusHandler(req.config);\n    const jobFunction = triggers.getJob(jobName, applicationId);\n    if (!jobFunction) {\n      throw new Parse.Error(Parse.Error.SCRIPT_FAILED, 'Invalid job.');\n    }\n    let params = Object.assign({}, req.body, req.query);\n    params = parseParams(params, req.config);\n    const request = {\n      params: params,\n      log: req.config.loggerController,\n      headers: req.config.headers,\n      ip: req.config.ip,\n      jobName,\n      message: jobHandler.setMessage.bind(jobHandler),\n    };\n\n    return jobHandler.setRunning(jobName, params).then(jobStatus => {\n      request.jobId = jobStatus.objectId;\n      // run the function async\n      process.nextTick(() => {\n        Promise.resolve()\n          .then(() => {\n            return jobFunction(request);\n          })\n          .then(\n            result => {\n              jobHandler.setSucceeded(result);\n            },\n            error => {\n              jobHandler.setFailed(error);\n            }\n          );\n      });\n      return {\n        headers: {\n          'X-Parse-Job-Status-Id': jobStatus.objectId,\n        },\n        response: {},\n      };\n    });\n  }\n\n  static createResponseObject(resolve, reject) {\n    return {\n      success: function (result) {\n        resolve({\n          response: {\n            result: Parse._encode(result),\n          },\n        });\n      },\n      error: function (message) {\n        const error = triggers.resolveError(message);\n        reject(error);\n      },\n    };\n  }\n  static handleCloudFunction(req) {\n    const functionName = req.params.functionName;\n    const applicationId = req.config.applicationId;\n    const theFunction = triggers.getFunction(functionName, applicationId);\n\n    if (!theFunction) {\n      throw new Parse.Error(Parse.Error.SCRIPT_FAILED, `Invalid function: \"${functionName}\"`);\n    }\n    let params = Object.assign({}, req.body, req.query);\n    params = parseParams(params, req.config);\n    const request = {\n      params: params,\n      master: req.auth && req.auth.isMaster,\n      user: req.auth && req.auth.user,\n      installationId: req.info.installationId,\n      log: req.config.loggerController,\n      headers: req.config.headers,\n      ip: req.config.ip,\n      functionName,\n      context: req.info.context,\n    };\n\n    return new Promise(function (resolve, reject) {\n      const userString = req.auth && req.auth.user ? req.auth.user.id : undefined;\n      const cleanInput = logger.truncateLogMessage(JSON.stringify(params));\n      const { success, error } = FunctionsRouter.createResponseObject(\n        result => {\n          try {\n            const cleanResult = logger.truncateLogMessage(JSON.stringify(result.response.result));\n            logger[req.config.logLevels.cloudFunctionSuccess](\n              `Ran cloud function ${functionName} for user ${userString} with:\\n  Input: ${cleanInput}\\n  Result: ${cleanResult}`,\n              {\n                functionName,\n                params,\n                user: userString,\n              }\n            );\n            resolve(result);\n          } catch (e) {\n            reject(e);\n          }\n        },\n        error => {\n          try {\n            logger[req.config.logLevels.cloudFunctionError](\n              `Failed running cloud function ${functionName} for user ${userString} with:\\n  Input: ${cleanInput}\\n  Error: ` +\n                JSON.stringify(error),\n              {\n                functionName,\n                error,\n                params,\n                user: userString,\n              }\n            );\n            reject(error);\n          } catch (e) {\n            reject(e);\n          }\n        }\n      );\n      return Promise.resolve()\n        .then(() => {\n          return triggers.maybeRunValidator(request, functionName, req.auth);\n        })\n        .then(() => {\n          return theFunction(request);\n        })\n        .then(success, error);\n    });\n  }\n}\n"],"mappings":";;;;;;AAKA;AACA;AACA;AACA;AACA;AAAmC;AATnC;;AAEA,IAAIA,KAAK,GAAGC,OAAO,CAAC,YAAY,CAAC,CAACD,KAAK;EACrCE,QAAQ,GAAGD,OAAO,CAAC,aAAa,CAAC;AAQnC,SAASE,WAAW,CAACC,GAAG,EAAEC,MAAM,EAAE;EAChC,IAAIC,KAAK,CAACC,OAAO,CAACH,GAAG,CAAC,EAAE;IACtB,OAAOA,GAAG,CAACI,GAAG,CAACC,IAAI,IAAI;MACrB,OAAON,WAAW,CAACM,IAAI,CAAC;IAC1B,CAAC,CAAC;EACJ,CAAC,MAAM,IAAIL,GAAG,IAAIA,GAAG,CAACM,MAAM,IAAI,MAAM,EAAE;IACtC,OAAOC,MAAM,CAACC,MAAM,CAAC,IAAIC,IAAI,CAACT,GAAG,CAACU,GAAG,CAAC,EAAEV,GAAG,CAAC;EAC9C,CAAC,MAAM,IAAIA,GAAG,IAAIA,GAAG,CAACM,MAAM,IAAI,MAAM,EAAE;IACtC,OAAOV,KAAK,CAACe,IAAI,CAACC,QAAQ,CAACZ,GAAG,CAAC;EACjC,CAAC,MAAM,IAAIA,GAAG,IAAIA,GAAG,CAACM,MAAM,IAAI,SAAS,IAAIL,MAAM,CAACY,gCAAgC,EAAE;IACpF,OAAOjB,KAAK,CAACW,MAAM,CAACK,QAAQ,CAAC;MAC3BN,MAAM,EAAE,SAAS;MACjBQ,SAAS,EAAEd,GAAG,CAACc,SAAS;MACxBC,QAAQ,EAAEf,GAAG,CAACe;IAChB,CAAC,CAAC;EACJ,CAAC,MAAM,IAAIf,GAAG,IAAI,OAAOA,GAAG,KAAK,QAAQ,EAAE;IACzC,OAAOgB,WAAW,CAAChB,GAAG,EAAEC,MAAM,CAAC;EACjC,CAAC,MAAM;IACL,OAAOD,GAAG;EACZ;AACF;AAEA,SAASgB,WAAW,CAACC,MAAM,EAAEhB,MAAM,EAAE;EACnC,OAAOiB,eAAC,CAACC,SAAS,CAACF,MAAM,EAAEZ,IAAI,IAAIN,WAAW,CAACM,IAAI,EAAEJ,MAAM,CAAC,CAAC;AAC/D;AAEO,MAAMmB,eAAe,SAASC,sBAAa,CAAC;EACjDC,WAAW,GAAG;IACZ,IAAI,CAACC,KAAK,CACR,MAAM,EACN,0BAA0B,EAC1BC,qCAAwB,EACxBJ,eAAe,CAACK,mBAAmB,CACpC;IACD,IAAI,CAACF,KAAK,CACR,MAAM,EACN,gBAAgB,EAChBC,qCAAwB,EACxBE,0CAA6B,EAC7B,UAAUC,GAAG,EAAE;MACb,OAAOP,eAAe,CAACQ,cAAc,CAACD,GAAG,CAAC;IAC5C,CAAC,CACF;IACD,IAAI,CAACJ,KAAK,CAAC,MAAM,EAAE,OAAO,EAAEG,0CAA6B,EAAE,UAAUC,GAAG,EAAE;MACxE,OAAOP,eAAe,CAACQ,cAAc,CAACD,GAAG,CAAC;IAC5C,CAAC,CAAC;EACJ;EAEA,OAAOC,cAAc,CAACD,GAAG,EAAE;IACzB,MAAME,OAAO,GAAGF,GAAG,CAACV,MAAM,CAACY,OAAO,IAAIF,GAAG,CAACG,IAAI,CAACD,OAAO;IACtD,MAAME,aAAa,GAAGJ,GAAG,CAAC1B,MAAM,CAAC8B,aAAa;IAC9C,MAAMC,UAAU,GAAG,IAAAC,+BAAgB,EAACN,GAAG,CAAC1B,MAAM,CAAC;IAC/C,MAAMiC,WAAW,GAAGpC,QAAQ,CAACqC,MAAM,CAACN,OAAO,EAAEE,aAAa,CAAC;IAC3D,IAAI,CAACG,WAAW,EAAE;MAChB,MAAM,IAAItC,KAAK,CAACwC,KAAK,CAACxC,KAAK,CAACwC,KAAK,CAACC,aAAa,EAAE,cAAc,CAAC;IAClE;IACA,IAAIpB,MAAM,GAAGV,MAAM,CAACC,MAAM,CAAC,CAAC,CAAC,EAAEmB,GAAG,CAACG,IAAI,EAAEH,GAAG,CAACW,KAAK,CAAC;IACnDrB,MAAM,GAAGD,WAAW,CAACC,MAAM,EAAEU,GAAG,CAAC1B,MAAM,CAAC;IACxC,MAAMsC,OAAO,GAAG;MACdtB,MAAM,EAAEA,MAAM;MACduB,GAAG,EAAEb,GAAG,CAAC1B,MAAM,CAACwC,gBAAgB;MAChCC,OAAO,EAAEf,GAAG,CAAC1B,MAAM,CAACyC,OAAO;MAC3BC,EAAE,EAAEhB,GAAG,CAAC1B,MAAM,CAAC0C,EAAE;MACjBd,OAAO;MACPe,OAAO,EAAEZ,UAAU,CAACa,UAAU,CAACC,IAAI,CAACd,UAAU;IAChD,CAAC;IAED,OAAOA,UAAU,CAACe,UAAU,CAAClB,OAAO,EAAEZ,MAAM,CAAC,CAAC+B,IAAI,CAACC,SAAS,IAAI;MAC9DV,OAAO,CAACW,KAAK,GAAGD,SAAS,CAAClC,QAAQ;MAClC;MACAoC,OAAO,CAACC,QAAQ,CAAC,MAAM;QACrBC,OAAO,CAACC,OAAO,EAAE,CACdN,IAAI,CAAC,MAAM;UACV,OAAOd,WAAW,CAACK,OAAO,CAAC;QAC7B,CAAC,CAAC,CACDS,IAAI,CACHO,MAAM,IAAI;UACRvB,UAAU,CAACwB,YAAY,CAACD,MAAM,CAAC;QACjC,CAAC,EACDE,KAAK,IAAI;UACPzB,UAAU,CAAC0B,SAAS,CAACD,KAAK,CAAC;QAC7B,CAAC,CACF;MACL,CAAC,CAAC;MACF,OAAO;QACLf,OAAO,EAAE;UACP,uBAAuB,EAAEO,SAAS,CAAClC;QACrC,CAAC;QACD4C,QAAQ,EAAE,CAAC;MACb,CAAC;IACH,CAAC,CAAC;EACJ;EAEA,OAAOC,oBAAoB,CAACN,OAAO,EAAEO,MAAM,EAAE;IAC3C,OAAO;MACLC,OAAO,EAAE,UAAUP,MAAM,EAAE;QACzBD,OAAO,CAAC;UACNK,QAAQ,EAAE;YACRJ,MAAM,EAAE3D,KAAK,CAACmE,OAAO,CAACR,MAAM;UAC9B;QACF,CAAC,CAAC;MACJ,CAAC;MACDE,KAAK,EAAE,UAAUb,OAAO,EAAE;QACxB,MAAMa,KAAK,GAAG3D,QAAQ,CAACkE,YAAY,CAACpB,OAAO,CAAC;QAC5CiB,MAAM,CAACJ,KAAK,CAAC;MACf;IACF,CAAC;EACH;EACA,OAAOhC,mBAAmB,CAACE,GAAG,EAAE;IAC9B,MAAMsC,YAAY,GAAGtC,GAAG,CAACV,MAAM,CAACgD,YAAY;IAC5C,MAAMlC,aAAa,GAAGJ,GAAG,CAAC1B,MAAM,CAAC8B,aAAa;IAC9C,MAAMmC,WAAW,GAAGpE,QAAQ,CAACqE,WAAW,CAACF,YAAY,EAAElC,aAAa,CAAC;IAErE,IAAI,CAACmC,WAAW,EAAE;MAChB,MAAM,IAAItE,KAAK,CAACwC,KAAK,CAACxC,KAAK,CAACwC,KAAK,CAACC,aAAa,EAAG,sBAAqB4B,YAAa,GAAE,CAAC;IACzF;IACA,IAAIhD,MAAM,GAAGV,MAAM,CAACC,MAAM,CAAC,CAAC,CAAC,EAAEmB,GAAG,CAACG,IAAI,EAAEH,GAAG,CAACW,KAAK,CAAC;IACnDrB,MAAM,GAAGD,WAAW,CAACC,MAAM,EAAEU,GAAG,CAAC1B,MAAM,CAAC;IACxC,MAAMsC,OAAO,GAAG;MACdtB,MAAM,EAAEA,MAAM;MACdmD,MAAM,EAAEzC,GAAG,CAAC0C,IAAI,IAAI1C,GAAG,CAAC0C,IAAI,CAACC,QAAQ;MACrCC,IAAI,EAAE5C,GAAG,CAAC0C,IAAI,IAAI1C,GAAG,CAAC0C,IAAI,CAACE,IAAI;MAC/BC,cAAc,EAAE7C,GAAG,CAAC8C,IAAI,CAACD,cAAc;MACvChC,GAAG,EAAEb,GAAG,CAAC1B,MAAM,CAACwC,gBAAgB;MAChCC,OAAO,EAAEf,GAAG,CAAC1B,MAAM,CAACyC,OAAO;MAC3BC,EAAE,EAAEhB,GAAG,CAAC1B,MAAM,CAAC0C,EAAE;MACjBsB,YAAY;MACZS,OAAO,EAAE/C,GAAG,CAAC8C,IAAI,CAACC;IACpB,CAAC;IAED,OAAO,IAAIrB,OAAO,CAAC,UAAUC,OAAO,EAAEO,MAAM,EAAE;MAC5C,MAAMc,UAAU,GAAGhD,GAAG,CAAC0C,IAAI,IAAI1C,GAAG,CAAC0C,IAAI,CAACE,IAAI,GAAG5C,GAAG,CAAC0C,IAAI,CAACE,IAAI,CAACK,EAAE,GAAGC,SAAS;MAC3E,MAAMC,UAAU,GAAGC,cAAM,CAACC,kBAAkB,CAACC,IAAI,CAACC,SAAS,CAACjE,MAAM,CAAC,CAAC;MACpE,MAAM;QAAE6C,OAAO;QAAEL;MAAM,CAAC,GAAGrC,eAAe,CAACwC,oBAAoB,CAC7DL,MAAM,IAAI;QACR,IAAI;UACF,MAAM4B,WAAW,GAAGJ,cAAM,CAACC,kBAAkB,CAACC,IAAI,CAACC,SAAS,CAAC3B,MAAM,CAACI,QAAQ,CAACJ,MAAM,CAAC,CAAC;UACrFwB,cAAM,CAACpD,GAAG,CAAC1B,MAAM,CAACmF,SAAS,CAACC,oBAAoB,CAAC,CAC9C,sBAAqBpB,YAAa,aAAYU,UAAW,oBAAmBG,UAAW,eAAcK,WAAY,EAAC,EACnH;YACElB,YAAY;YACZhD,MAAM;YACNsD,IAAI,EAAEI;UACR,CAAC,CACF;UACDrB,OAAO,CAACC,MAAM,CAAC;QACjB,CAAC,CAAC,OAAO+B,CAAC,EAAE;UACVzB,MAAM,CAACyB,CAAC,CAAC;QACX;MACF,CAAC,EACD7B,KAAK,IAAI;QACP,IAAI;UACFsB,cAAM,CAACpD,GAAG,CAAC1B,MAAM,CAACmF,SAAS,CAACG,kBAAkB,CAAC,CAC5C,iCAAgCtB,YAAa,aAAYU,UAAW,oBAAmBG,UAAW,aAAY,GAC7GG,IAAI,CAACC,SAAS,CAACzB,KAAK,CAAC,EACvB;YACEQ,YAAY;YACZR,KAAK;YACLxC,MAAM;YACNsD,IAAI,EAAEI;UACR,CAAC,CACF;UACDd,MAAM,CAACJ,KAAK,CAAC;QACf,CAAC,CAAC,OAAO6B,CAAC,EAAE;UACVzB,MAAM,CAACyB,CAAC,CAAC;QACX;MACF,CAAC,CACF;MACD,OAAOjC,OAAO,CAACC,OAAO,EAAE,CACrBN,IAAI,CAAC,MAAM;QACV,OAAOlD,QAAQ,CAAC0F,iBAAiB,CAACjD,OAAO,EAAE0B,YAAY,EAAEtC,GAAG,CAAC0C,IAAI,CAAC;MACpE,CAAC,CAAC,CACDrB,IAAI,CAAC,MAAM;QACV,OAAOkB,WAAW,CAAC3B,OAAO,CAAC;MAC7B,CAAC,CAAC,CACDS,IAAI,CAACc,OAAO,EAAEL,KAAK,CAAC;IACzB,CAAC,CAAC;EACJ;AACF;AAAC"}
@@ -122,7 +122,7 @@ class PagesRouter extends _PromiseRouter.default {
122
122
  return this.goToPage(req, pages.emailVerificationLinkInvalid);
123
123
  }
124
124
  const userController = config.userController;
125
- return userController.resendVerificationEmail(username).then(() => {
125
+ return userController.resendVerificationEmail(username, req).then(() => {
126
126
  return this.goToPage(req, pages.emailVerificationSendSuccess);
127
127
  }, () => {
128
128
  return this.goToPage(req, pages.emailVerificationSendFail);
@@ -659,4 +659,4 @@ module.exports = {
659
659
  pageParams,
660
660
  pages
661
661
  };
662
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["pages","Object","freeze","passwordReset","Page","id","defaultFile","passwordResetSuccess","passwordResetLinkInvalid","emailVerificationSuccess","emailVerificationSendFail","emailVerificationSendSuccess","emailVerificationLinkInvalid","emailVerificationLinkExpired","pageParams","appName","appId","token","username","error","locale","publicServerUrl","pageParamHeaderPrefix","errors","jsonFailedFileLoading","fileOutsideAllowedScope","PagesRouter","PromiseRouter","constructor","pagesConfig","pagesEndpoint","pagesPath","path","resolve","__dirname","loadJsonResource","mountPagesRoutes","mountCustomRoutes","mountStaticRoute","verifyEmail","req","config","rawToken","query","toString","invalidRequest","goToPage","userController","then","params","resendVerificationEmail","body","publicServerURL","requestResetPassword","checkResetTokenValidity","applicationId","resetPassword","new_password","xhr","Parse","Error","USERNAME_MISSING","OTHER_CAUSE","PASSWORD_MISSING","updatePassword","Promise","success","err","result","status","response","page","responseType","redirect","forceRedirect","undefined","method","defaultParams","getDefaultParams","values","includes","notFound","assign","getLocale","defaultPath","defaultPagePath","defaultUrl","composePageUrl","customUrl","customUrls","Utils","isPath","redirectResponse","placeholders","enableLocalization","localizationJsonPath","getJsonPlaceholders","getLocalizedPath","subdir","pageResponse","staticRoute","relativePath","absolutePath","endsWith","fileResponse","getJsonTranslation","jsonParameters","localizationFallbackLocale","language","split","resource","translation","JSON","stringify","mustache","render","parse","data","readFile","e","configPlaceholders","prototype","call","allPlaceholders","paramsAndPlaceholders","headers","entries","reduce","m","p","toLowerCase","text","filePath","normalizedPath","normalize","startsWith","fs","json","require","url","location","URL","forEach","searchParams","set","locationString","file","join","message","setConfig","failGracefully","Config","get","route","customRoutes","handler","expressRouter","router","express","Router","use","module","exports"],"sources":["../../src/Routers/PagesRouter.js"],"sourcesContent":["import PromiseRouter from '../PromiseRouter';\nimport Config from '../Config';\nimport express from 'express';\nimport path from 'path';\nimport { promises as fs } from 'fs';\nimport { Parse } from 'parse/node';\nimport Utils from '../Utils';\nimport mustache from 'mustache';\nimport Page from '../Page';\n\n// All pages with custom page key for reference and file name\nconst pages = Object.freeze({\n  passwordReset: new Page({ id: 'passwordReset', defaultFile: 'password_reset.html' }),\n  passwordResetSuccess: new Page({\n    id: 'passwordResetSuccess',\n    defaultFile: 'password_reset_success.html',\n  }),\n  passwordResetLinkInvalid: new Page({\n    id: 'passwordResetLinkInvalid',\n    defaultFile: 'password_reset_link_invalid.html',\n  }),\n  emailVerificationSuccess: new Page({\n    id: 'emailVerificationSuccess',\n    defaultFile: 'email_verification_success.html',\n  }),\n  emailVerificationSendFail: new Page({\n    id: 'emailVerificationSendFail',\n    defaultFile: 'email_verification_send_fail.html',\n  }),\n  emailVerificationSendSuccess: new Page({\n    id: 'emailVerificationSendSuccess',\n    defaultFile: 'email_verification_send_success.html',\n  }),\n  emailVerificationLinkInvalid: new Page({\n    id: 'emailVerificationLinkInvalid',\n    defaultFile: 'email_verification_link_invalid.html',\n  }),\n  emailVerificationLinkExpired: new Page({\n    id: 'emailVerificationLinkExpired',\n    defaultFile: 'email_verification_link_expired.html',\n  }),\n});\n\n// All page parameters for reference to be used as template placeholders or query params\nconst pageParams = Object.freeze({\n  appName: 'appName',\n  appId: 'appId',\n  token: 'token',\n  username: 'username',\n  error: 'error',\n  locale: 'locale',\n  publicServerUrl: 'publicServerUrl',\n});\n\n// The header prefix to add page params as response headers\nconst pageParamHeaderPrefix = 'x-parse-page-param-';\n\n// The errors being thrown\nconst errors = Object.freeze({\n  jsonFailedFileLoading: 'failed to load JSON file',\n  fileOutsideAllowedScope: 'not allowed to read file outside of pages directory',\n});\n\nexport class PagesRouter extends PromiseRouter {\n  /**\n   * Constructs a PagesRouter.\n   * @param {Object} pages The pages options from the Parse Server configuration.\n   */\n  constructor(pages = {}) {\n    super();\n\n    // Set instance properties\n    this.pagesConfig = pages;\n    this.pagesEndpoint = pages.pagesEndpoint ? pages.pagesEndpoint : 'apps';\n    this.pagesPath = pages.pagesPath\n      ? path.resolve('./', pages.pagesPath)\n      : path.resolve(__dirname, '../../public');\n    this.loadJsonResource();\n    this.mountPagesRoutes();\n    this.mountCustomRoutes();\n    this.mountStaticRoute();\n  }\n\n  verifyEmail(req) {\n    const config = req.config;\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!token || !username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n    return userController.verifyEmail(username, token).then(\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationSuccess, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationLinkExpired, params);\n      }\n    );\n  }\n\n  resendVerificationEmail(req) {\n    const config = req.config;\n    const username = req.body.username;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n\n    return userController.resendVerificationEmail(username).then(\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendSuccess);\n      },\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendFail);\n      }\n    );\n  }\n\n  passwordReset(req) {\n    const config = req.config;\n    const params = {\n      [pageParams.appId]: req.params.appId,\n      [pageParams.appName]: config.appName,\n      [pageParams.token]: req.query.token,\n      [pageParams.username]: req.query.username,\n      [pageParams.publicServerUrl]: config.publicServerURL,\n    };\n    return this.goToPage(req, pages.passwordReset, params);\n  }\n\n  requestResetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!username || !token) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    return config.userController.checkResetTokenValidity(username, token).then(\n      () => {\n        const params = {\n          [pageParams.token]: token,\n          [pageParams.username]: username,\n          [pageParams.appId]: config.applicationId,\n          [pageParams.appName]: config.appName,\n        };\n        return this.goToPage(req, pages.passwordReset, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.passwordResetLinkInvalid, params);\n      }\n    );\n  }\n\n  resetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, new_password, token: rawToken } = req.body;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if ((!username || !token || !new_password) && req.xhr === false) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    if (!username) {\n      throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'Missing username');\n    }\n\n    if (!token) {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'Missing token');\n    }\n\n    if (!new_password) {\n      throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'Missing password');\n    }\n\n    return config.userController\n      .updatePassword(username, token, new_password)\n      .then(\n        () => {\n          return Promise.resolve({\n            success: true,\n          });\n        },\n        err => {\n          return Promise.resolve({\n            success: false,\n            err,\n          });\n        }\n      )\n      .then(result => {\n        if (req.xhr) {\n          if (result.success) {\n            return Promise.resolve({\n              status: 200,\n              response: 'Password successfully reset',\n            });\n          }\n          if (result.err) {\n            throw new Parse.Error(Parse.Error.OTHER_CAUSE, `${result.err}`);\n          }\n        }\n\n        const query = result.success\n          ? {\n            [pageParams.username]: username,\n          }\n          : {\n            [pageParams.username]: username,\n            [pageParams.token]: token,\n            [pageParams.appId]: config.applicationId,\n            [pageParams.error]: result.err,\n            [pageParams.appName]: config.appName,\n          };\n        const page = result.success ? pages.passwordResetSuccess : pages.passwordReset;\n\n        return this.goToPage(req, page, query, false);\n      });\n  }\n\n  /**\n   * Returns page content if the page is a local file or returns a\n   * redirect to a custom page.\n   * @param {Object} req The express request.\n   * @param {Page} page The page to go to.\n   * @param {Object} [params={}] The query parameters to attach to the URL in case of\n   * HTTP redirect responses for POST requests, or the placeholders to fill into\n   * the response content in case of HTTP content responses for GET requests.\n   * @param {Boolean} [responseType] Is true if a redirect response should be forced,\n   * false if a content response should be forced, undefined if the response type\n   * should depend on the request type by default:\n   * - GET request -> content response\n   * - POST request -> redirect response (PRG pattern)\n   * @returns {Promise<Object>} The PromiseRouter response.\n   */\n  goToPage(req, page, params = {}, responseType) {\n    const config = req.config;\n\n    // Determine redirect either by force, response setting or request method\n    const redirect = config.pages.forceRedirect\n      ? true\n      : responseType !== undefined\n        ? responseType\n        : req.method == 'POST';\n\n    // Include default parameters\n    const defaultParams = this.getDefaultParams(config);\n    if (Object.values(defaultParams).includes(undefined)) {\n      return this.notFound();\n    }\n    params = Object.assign(params, defaultParams);\n\n    // Add locale to params to ensure it is passed on with every request;\n    // that means, once a locale is set, it is passed on to any follow-up page,\n    // e.g. request_password_reset -> password_reset -> password_reset_success\n    const locale = this.getLocale(req);\n    params[pageParams.locale] = locale;\n\n    // Compose paths and URLs\n    const defaultFile = page.defaultFile;\n    const defaultPath = this.defaultPagePath(defaultFile);\n    const defaultUrl = this.composePageUrl(defaultFile, config.publicServerURL);\n\n    // If custom URL is set redirect to it without localization\n    const customUrl = config.pages.customUrls[page.id];\n    if (customUrl && !Utils.isPath(customUrl)) {\n      return this.redirectResponse(customUrl, params);\n    }\n\n    // Get JSON placeholders\n    let placeholders = {};\n    if (config.pages.enableLocalization && config.pages.localizationJsonPath) {\n      placeholders = this.getJsonPlaceholders(locale, params);\n    }\n\n    // Send response\n    if (config.pages.enableLocalization && locale) {\n      return Utils.getLocalizedPath(defaultPath, locale).then(({ path, subdir }) =>\n        redirect\n          ? this.redirectResponse(\n            this.composePageUrl(defaultFile, config.publicServerURL, subdir),\n            params\n          )\n          : this.pageResponse(path, params, placeholders)\n      );\n    } else {\n      return redirect\n        ? this.redirectResponse(defaultUrl, params)\n        : this.pageResponse(defaultPath, params, placeholders);\n    }\n  }\n\n  /**\n   * Serves a request to a static resource and localizes the resource if it\n   * is a HTML file.\n   * @param {Object} req The request object.\n   * @returns {Promise<Object>} The response.\n   */\n  staticRoute(req) {\n    // Get requested path\n    const relativePath = req.params[0];\n\n    // Resolve requested path to absolute path\n    const absolutePath = path.resolve(this.pagesPath, relativePath);\n\n    // If the requested file is not a HTML file send its raw content\n    if (!absolutePath || !absolutePath.endsWith('.html')) {\n      return this.fileResponse(absolutePath);\n    }\n\n    // Get parameters\n    const params = this.getDefaultParams(req.config);\n    const locale = this.getLocale(req);\n    if (locale) {\n      params.locale = locale;\n    }\n\n    // Get JSON placeholders\n    const placeholders = this.getJsonPlaceholders(locale, params);\n\n    return this.pageResponse(absolutePath, params, placeholders);\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale. The JSON\n   * resource is parsed according to i18next syntax.\n   *\n   * Example JSON content:\n   * ```js\n   *  {\n   *    \"en\": {               // resource for language `en` (English)\n   *      \"translation\": {\n   *        \"greeting\": \"Hello!\"\n   *      }\n   *    },\n   *    \"de\": {               // resource for language `de` (German)\n   *      \"translation\": {\n   *        \"greeting\": \"Hallo!\"\n   *      }\n   *    }\n   *    \"de-CH\": {            // resource for locale `de-CH` (Swiss German)\n   *      \"translation\": {\n   *        \"greeting\": \"Grüezi!\"\n   *      }\n   *    }\n   *  }\n   * ```\n   * @param {String} locale The locale to translate to.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonTranslation(locale) {\n    // If there is no JSON resource\n    if (this.jsonParameters === undefined) {\n      return {};\n    }\n\n    // If locale is not set use the fallback locale\n    locale = locale || this.pagesConfig.localizationFallbackLocale;\n\n    // Get matching translation by locale, language or fallback locale\n    const language = locale.split('-')[0];\n    const resource =\n      this.jsonParameters[locale] ||\n      this.jsonParameters[language] ||\n      this.jsonParameters[this.pagesConfig.localizationFallbackLocale] ||\n      {};\n    const translation = resource.translation || {};\n    return translation;\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale with\n   * placeholders filled in by given parameters.\n   * @param {String} locale The locale to translate to.\n   * @param {Object} params The parameters to fill into any placeholders\n   * within the translations.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonPlaceholders(locale, params = {}) {\n    // If localization is disabled or there is no JSON resource\n    if (!this.pagesConfig.enableLocalization || !this.pagesConfig.localizationJsonPath) {\n      return {};\n    }\n\n    // Get JSON placeholders\n    let placeholders = this.getJsonTranslation(locale);\n\n    // Fill in any placeholders in the translation; this allows a translation\n    // to contain default placeholders like {{appName}} which are filled here\n    placeholders = JSON.stringify(placeholders);\n    placeholders = mustache.render(placeholders, params);\n    placeholders = JSON.parse(placeholders);\n\n    return placeholders;\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @param {Object} [params={}] The parameters to be included in the response\n   * header. These will also be used to fill placeholders.\n   * @param {Object} [placeholders={}] The placeholders to fill in the content.\n   * These will not be included in the response header.\n   * @returns {Object} The Promise Router response.\n   */\n  async pageResponse(path, params = {}, placeholders = {}) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    // Get config placeholders; can be an object, a function or an async function\n    let configPlaceholders =\n      typeof this.pagesConfig.placeholders === 'function'\n        ? this.pagesConfig.placeholders(params)\n        : Object.prototype.toString.call(this.pagesConfig.placeholders) === '[object Object]'\n          ? this.pagesConfig.placeholders\n          : {};\n    if (configPlaceholders instanceof Promise) {\n      configPlaceholders = await configPlaceholders;\n    }\n\n    // Fill placeholders\n    const allPlaceholders = Object.assign({}, configPlaceholders, placeholders);\n    const paramsAndPlaceholders = Object.assign({}, params, allPlaceholders);\n    data = mustache.render(data, paramsAndPlaceholders);\n\n    // Add placeholders in header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return { text: data, headers: headers };\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @returns {Object} The PromiseRouter response.\n   */\n  async fileResponse(path) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    return { text: data };\n  }\n\n  /**\n   * Reads and returns the content of a file at a given path. File reading to\n   * serve content on the static route is only allowed from the pages\n   * directory on downwards.\n   * -----------------------------------------------------------------------\n   * **WARNING:** All file reads in the PagesRouter must be executed by this\n   * wrapper because it also detects and prevents common exploits.\n   * -----------------------------------------------------------------------\n   * @param {String} filePath The path to the file to read.\n   * @returns {Promise<String>} The file content.\n   */\n  async readFile(filePath) {\n    // Normalize path to prevent it from containing any directory changing\n    // UNIX patterns which could expose the whole file system, e.g.\n    // `http://example.com/parse/apps/../file.txt` requests a file outside\n    // of the pages directory scope.\n    const normalizedPath = path.normalize(filePath);\n\n    // Abort if the path is outside of the path directory scope\n    if (!normalizedPath.startsWith(this.pagesPath)) {\n      throw errors.fileOutsideAllowedScope;\n    }\n\n    return await fs.readFile(normalizedPath, 'utf-8');\n  }\n\n  /**\n   * Loads a language resource JSON file that is used for translations.\n   */\n  loadJsonResource() {\n    if (this.pagesConfig.localizationJsonPath === undefined) {\n      return;\n    }\n    try {\n      const json = require(path.resolve('./', this.pagesConfig.localizationJsonPath));\n      this.jsonParameters = json;\n    } catch (e) {\n      throw errors.jsonFailedFileLoading;\n    }\n  }\n\n  /**\n   * Extracts and returns the page default parameters from the Parse Server\n   * configuration. These parameters are made accessible in every page served\n   * by this router.\n   * @param {Object} config The Parse Server configuration.\n   * @returns {Object} The default parameters.\n   */\n  getDefaultParams(config) {\n    return config\n      ? {\n        [pageParams.appId]: config.appId,\n        [pageParams.appName]: config.appName,\n        [pageParams.publicServerUrl]: config.publicServerURL,\n      }\n      : {};\n  }\n\n  /**\n   * Extracts and returns the locale from an express request.\n   * @param {Object} req The express request.\n   * @returns {String|undefined} The locale, or undefined if no locale was set.\n   */\n  getLocale(req) {\n    const locale =\n      (req.query || {})[pageParams.locale] ||\n      (req.body || {})[pageParams.locale] ||\n      (req.params || {})[pageParams.locale] ||\n      (req.headers || {})[pageParamHeaderPrefix + pageParams.locale];\n    return locale;\n  }\n\n  /**\n   * Creates a response with http redirect.\n   * @param {Object} req The express request.\n   * @param {String} path The path of the file to return.\n   * @param {Object} params The query parameters to include.\n   * @returns {Object} The Promise Router response.\n   */\n  async redirectResponse(url, params) {\n    // Remove any parameters with undefined value\n    params = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[p[0]] = p[1];\n      }\n      return m;\n    }, {});\n\n    // Compose URL with parameters in query\n    const location = new URL(url);\n    Object.entries(params).forEach(p => location.searchParams.set(p[0], p[1]));\n    const locationString = location.toString();\n\n    // Add parameters to header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return {\n      status: 303,\n      location: locationString,\n      headers: headers,\n    };\n  }\n\n  defaultPagePath(file) {\n    return path.join(this.pagesPath, file);\n  }\n\n  composePageUrl(file, publicServerUrl, locale) {\n    let url = publicServerUrl;\n    url += url.endsWith('/') ? '' : '/';\n    url += this.pagesEndpoint + '/';\n    url += locale === undefined ? '' : locale + '/';\n    url += file;\n    return url;\n  }\n\n  notFound() {\n    return {\n      text: 'Not found.',\n      status: 404,\n    };\n  }\n\n  invalidRequest() {\n    const error = new Error();\n    error.status = 403;\n    error.message = 'unauthorized';\n    throw error;\n  }\n\n  /**\n   * Sets the Parse Server configuration in the request object to make it\n   * easily accessible throughtout request processing.\n   * @param {Object} req The request.\n   * @param {Boolean} failGracefully Is true if failing to set the config should\n   * not result in an invalid request response. Default is `false`.\n   */\n  setConfig(req, failGracefully = false) {\n    req.config = Config.get(req.params.appId || req.query.appId);\n    if (!req.config && !failGracefully) {\n      this.invalidRequest();\n    }\n    return Promise.resolve();\n  }\n\n  mountPagesRoutes() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/verify_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.verifyEmail(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/resend_verification_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resendVerificationEmail(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/choose_password`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.passwordReset(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resetPassword(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.requestResetPassword(req);\n      }\n    );\n  }\n\n  mountCustomRoutes() {\n    for (const route of this.pagesConfig.customRoutes || []) {\n      this.route(\n        route.method,\n        `/${this.pagesEndpoint}/:appId/${route.path}`,\n        req => {\n          this.setConfig(req);\n        },\n        async req => {\n          const { file, query = {} } = (await route.handler(req)) || {};\n\n          // If route handler did not return a page send 404 response\n          if (!file) {\n            return this.notFound();\n          }\n\n          // Send page response\n          const page = new Page({ id: file, defaultFile: file });\n          return this.goToPage(req, page, query, false);\n        }\n      );\n    }\n  }\n\n  mountStaticRoute() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/(*)?`,\n      req => {\n        this.setConfig(req, true);\n      },\n      req => {\n        return this.staticRoute(req);\n      }\n    );\n  }\n\n  expressRouter() {\n    const router = express.Router();\n    router.use('/', super.expressRouter());\n    return router;\n  }\n}\n\nexport default PagesRouter;\nmodule.exports = {\n  PagesRouter,\n  pageParamHeaderPrefix,\n  pageParams,\n  pages,\n};\n"],"mappings":"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"}
662
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["pages","Object","freeze","passwordReset","Page","id","defaultFile","passwordResetSuccess","passwordResetLinkInvalid","emailVerificationSuccess","emailVerificationSendFail","emailVerificationSendSuccess","emailVerificationLinkInvalid","emailVerificationLinkExpired","pageParams","appName","appId","token","username","error","locale","publicServerUrl","pageParamHeaderPrefix","errors","jsonFailedFileLoading","fileOutsideAllowedScope","PagesRouter","PromiseRouter","constructor","pagesConfig","pagesEndpoint","pagesPath","path","resolve","__dirname","loadJsonResource","mountPagesRoutes","mountCustomRoutes","mountStaticRoute","verifyEmail","req","config","rawToken","query","toString","invalidRequest","goToPage","userController","then","params","resendVerificationEmail","body","publicServerURL","requestResetPassword","checkResetTokenValidity","applicationId","resetPassword","new_password","xhr","Parse","Error","USERNAME_MISSING","OTHER_CAUSE","PASSWORD_MISSING","updatePassword","Promise","success","err","result","status","response","page","responseType","redirect","forceRedirect","undefined","method","defaultParams","getDefaultParams","values","includes","notFound","assign","getLocale","defaultPath","defaultPagePath","defaultUrl","composePageUrl","customUrl","customUrls","Utils","isPath","redirectResponse","placeholders","enableLocalization","localizationJsonPath","getJsonPlaceholders","getLocalizedPath","subdir","pageResponse","staticRoute","relativePath","absolutePath","endsWith","fileResponse","getJsonTranslation","jsonParameters","localizationFallbackLocale","language","split","resource","translation","JSON","stringify","mustache","render","parse","data","readFile","e","configPlaceholders","prototype","call","allPlaceholders","paramsAndPlaceholders","headers","entries","reduce","m","p","toLowerCase","text","filePath","normalizedPath","normalize","startsWith","fs","json","require","url","location","URL","forEach","searchParams","set","locationString","file","join","message","setConfig","failGracefully","Config","get","route","customRoutes","handler","expressRouter","router","express","Router","use","module","exports"],"sources":["../../src/Routers/PagesRouter.js"],"sourcesContent":["import PromiseRouter from '../PromiseRouter';\nimport Config from '../Config';\nimport express from 'express';\nimport path from 'path';\nimport { promises as fs } from 'fs';\nimport { Parse } from 'parse/node';\nimport Utils from '../Utils';\nimport mustache from 'mustache';\nimport Page from '../Page';\n\n// All pages with custom page key for reference and file name\nconst pages = Object.freeze({\n  passwordReset: new Page({ id: 'passwordReset', defaultFile: 'password_reset.html' }),\n  passwordResetSuccess: new Page({\n    id: 'passwordResetSuccess',\n    defaultFile: 'password_reset_success.html',\n  }),\n  passwordResetLinkInvalid: new Page({\n    id: 'passwordResetLinkInvalid',\n    defaultFile: 'password_reset_link_invalid.html',\n  }),\n  emailVerificationSuccess: new Page({\n    id: 'emailVerificationSuccess',\n    defaultFile: 'email_verification_success.html',\n  }),\n  emailVerificationSendFail: new Page({\n    id: 'emailVerificationSendFail',\n    defaultFile: 'email_verification_send_fail.html',\n  }),\n  emailVerificationSendSuccess: new Page({\n    id: 'emailVerificationSendSuccess',\n    defaultFile: 'email_verification_send_success.html',\n  }),\n  emailVerificationLinkInvalid: new Page({\n    id: 'emailVerificationLinkInvalid',\n    defaultFile: 'email_verification_link_invalid.html',\n  }),\n  emailVerificationLinkExpired: new Page({\n    id: 'emailVerificationLinkExpired',\n    defaultFile: 'email_verification_link_expired.html',\n  }),\n});\n\n// All page parameters for reference to be used as template placeholders or query params\nconst pageParams = Object.freeze({\n  appName: 'appName',\n  appId: 'appId',\n  token: 'token',\n  username: 'username',\n  error: 'error',\n  locale: 'locale',\n  publicServerUrl: 'publicServerUrl',\n});\n\n// The header prefix to add page params as response headers\nconst pageParamHeaderPrefix = 'x-parse-page-param-';\n\n// The errors being thrown\nconst errors = Object.freeze({\n  jsonFailedFileLoading: 'failed to load JSON file',\n  fileOutsideAllowedScope: 'not allowed to read file outside of pages directory',\n});\n\nexport class PagesRouter extends PromiseRouter {\n  /**\n   * Constructs a PagesRouter.\n   * @param {Object} pages The pages options from the Parse Server configuration.\n   */\n  constructor(pages = {}) {\n    super();\n\n    // Set instance properties\n    this.pagesConfig = pages;\n    this.pagesEndpoint = pages.pagesEndpoint ? pages.pagesEndpoint : 'apps';\n    this.pagesPath = pages.pagesPath\n      ? path.resolve('./', pages.pagesPath)\n      : path.resolve(__dirname, '../../public');\n    this.loadJsonResource();\n    this.mountPagesRoutes();\n    this.mountCustomRoutes();\n    this.mountStaticRoute();\n  }\n\n  verifyEmail(req) {\n    const config = req.config;\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!token || !username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n    return userController.verifyEmail(username, token).then(\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationSuccess, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationLinkExpired, params);\n      }\n    );\n  }\n\n  resendVerificationEmail(req) {\n    const config = req.config;\n    const username = req.body.username;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n\n    return userController.resendVerificationEmail(username, req).then(\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendSuccess);\n      },\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendFail);\n      }\n    );\n  }\n\n  passwordReset(req) {\n    const config = req.config;\n    const params = {\n      [pageParams.appId]: req.params.appId,\n      [pageParams.appName]: config.appName,\n      [pageParams.token]: req.query.token,\n      [pageParams.username]: req.query.username,\n      [pageParams.publicServerUrl]: config.publicServerURL,\n    };\n    return this.goToPage(req, pages.passwordReset, params);\n  }\n\n  requestResetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!username || !token) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    return config.userController.checkResetTokenValidity(username, token).then(\n      () => {\n        const params = {\n          [pageParams.token]: token,\n          [pageParams.username]: username,\n          [pageParams.appId]: config.applicationId,\n          [pageParams.appName]: config.appName,\n        };\n        return this.goToPage(req, pages.passwordReset, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.passwordResetLinkInvalid, params);\n      }\n    );\n  }\n\n  resetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, new_password, token: rawToken } = req.body;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if ((!username || !token || !new_password) && req.xhr === false) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    if (!username) {\n      throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'Missing username');\n    }\n\n    if (!token) {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'Missing token');\n    }\n\n    if (!new_password) {\n      throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'Missing password');\n    }\n\n    return config.userController\n      .updatePassword(username, token, new_password)\n      .then(\n        () => {\n          return Promise.resolve({\n            success: true,\n          });\n        },\n        err => {\n          return Promise.resolve({\n            success: false,\n            err,\n          });\n        }\n      )\n      .then(result => {\n        if (req.xhr) {\n          if (result.success) {\n            return Promise.resolve({\n              status: 200,\n              response: 'Password successfully reset',\n            });\n          }\n          if (result.err) {\n            throw new Parse.Error(Parse.Error.OTHER_CAUSE, `${result.err}`);\n          }\n        }\n\n        const query = result.success\n          ? {\n            [pageParams.username]: username,\n          }\n          : {\n            [pageParams.username]: username,\n            [pageParams.token]: token,\n            [pageParams.appId]: config.applicationId,\n            [pageParams.error]: result.err,\n            [pageParams.appName]: config.appName,\n          };\n        const page = result.success ? pages.passwordResetSuccess : pages.passwordReset;\n\n        return this.goToPage(req, page, query, false);\n      });\n  }\n\n  /**\n   * Returns page content if the page is a local file or returns a\n   * redirect to a custom page.\n   * @param {Object} req The express request.\n   * @param {Page} page The page to go to.\n   * @param {Object} [params={}] The query parameters to attach to the URL in case of\n   * HTTP redirect responses for POST requests, or the placeholders to fill into\n   * the response content in case of HTTP content responses for GET requests.\n   * @param {Boolean} [responseType] Is true if a redirect response should be forced,\n   * false if a content response should be forced, undefined if the response type\n   * should depend on the request type by default:\n   * - GET request -> content response\n   * - POST request -> redirect response (PRG pattern)\n   * @returns {Promise<Object>} The PromiseRouter response.\n   */\n  goToPage(req, page, params = {}, responseType) {\n    const config = req.config;\n\n    // Determine redirect either by force, response setting or request method\n    const redirect = config.pages.forceRedirect\n      ? true\n      : responseType !== undefined\n        ? responseType\n        : req.method == 'POST';\n\n    // Include default parameters\n    const defaultParams = this.getDefaultParams(config);\n    if (Object.values(defaultParams).includes(undefined)) {\n      return this.notFound();\n    }\n    params = Object.assign(params, defaultParams);\n\n    // Add locale to params to ensure it is passed on with every request;\n    // that means, once a locale is set, it is passed on to any follow-up page,\n    // e.g. request_password_reset -> password_reset -> password_reset_success\n    const locale = this.getLocale(req);\n    params[pageParams.locale] = locale;\n\n    // Compose paths and URLs\n    const defaultFile = page.defaultFile;\n    const defaultPath = this.defaultPagePath(defaultFile);\n    const defaultUrl = this.composePageUrl(defaultFile, config.publicServerURL);\n\n    // If custom URL is set redirect to it without localization\n    const customUrl = config.pages.customUrls[page.id];\n    if (customUrl && !Utils.isPath(customUrl)) {\n      return this.redirectResponse(customUrl, params);\n    }\n\n    // Get JSON placeholders\n    let placeholders = {};\n    if (config.pages.enableLocalization && config.pages.localizationJsonPath) {\n      placeholders = this.getJsonPlaceholders(locale, params);\n    }\n\n    // Send response\n    if (config.pages.enableLocalization && locale) {\n      return Utils.getLocalizedPath(defaultPath, locale).then(({ path, subdir }) =>\n        redirect\n          ? this.redirectResponse(\n            this.composePageUrl(defaultFile, config.publicServerURL, subdir),\n            params\n          )\n          : this.pageResponse(path, params, placeholders)\n      );\n    } else {\n      return redirect\n        ? this.redirectResponse(defaultUrl, params)\n        : this.pageResponse(defaultPath, params, placeholders);\n    }\n  }\n\n  /**\n   * Serves a request to a static resource and localizes the resource if it\n   * is a HTML file.\n   * @param {Object} req The request object.\n   * @returns {Promise<Object>} The response.\n   */\n  staticRoute(req) {\n    // Get requested path\n    const relativePath = req.params[0];\n\n    // Resolve requested path to absolute path\n    const absolutePath = path.resolve(this.pagesPath, relativePath);\n\n    // If the requested file is not a HTML file send its raw content\n    if (!absolutePath || !absolutePath.endsWith('.html')) {\n      return this.fileResponse(absolutePath);\n    }\n\n    // Get parameters\n    const params = this.getDefaultParams(req.config);\n    const locale = this.getLocale(req);\n    if (locale) {\n      params.locale = locale;\n    }\n\n    // Get JSON placeholders\n    const placeholders = this.getJsonPlaceholders(locale, params);\n\n    return this.pageResponse(absolutePath, params, placeholders);\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale. The JSON\n   * resource is parsed according to i18next syntax.\n   *\n   * Example JSON content:\n   * ```js\n   *  {\n   *    \"en\": {               // resource for language `en` (English)\n   *      \"translation\": {\n   *        \"greeting\": \"Hello!\"\n   *      }\n   *    },\n   *    \"de\": {               // resource for language `de` (German)\n   *      \"translation\": {\n   *        \"greeting\": \"Hallo!\"\n   *      }\n   *    }\n   *    \"de-CH\": {            // resource for locale `de-CH` (Swiss German)\n   *      \"translation\": {\n   *        \"greeting\": \"Grüezi!\"\n   *      }\n   *    }\n   *  }\n   * ```\n   * @param {String} locale The locale to translate to.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonTranslation(locale) {\n    // If there is no JSON resource\n    if (this.jsonParameters === undefined) {\n      return {};\n    }\n\n    // If locale is not set use the fallback locale\n    locale = locale || this.pagesConfig.localizationFallbackLocale;\n\n    // Get matching translation by locale, language or fallback locale\n    const language = locale.split('-')[0];\n    const resource =\n      this.jsonParameters[locale] ||\n      this.jsonParameters[language] ||\n      this.jsonParameters[this.pagesConfig.localizationFallbackLocale] ||\n      {};\n    const translation = resource.translation || {};\n    return translation;\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale with\n   * placeholders filled in by given parameters.\n   * @param {String} locale The locale to translate to.\n   * @param {Object} params The parameters to fill into any placeholders\n   * within the translations.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonPlaceholders(locale, params = {}) {\n    // If localization is disabled or there is no JSON resource\n    if (!this.pagesConfig.enableLocalization || !this.pagesConfig.localizationJsonPath) {\n      return {};\n    }\n\n    // Get JSON placeholders\n    let placeholders = this.getJsonTranslation(locale);\n\n    // Fill in any placeholders in the translation; this allows a translation\n    // to contain default placeholders like {{appName}} which are filled here\n    placeholders = JSON.stringify(placeholders);\n    placeholders = mustache.render(placeholders, params);\n    placeholders = JSON.parse(placeholders);\n\n    return placeholders;\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @param {Object} [params={}] The parameters to be included in the response\n   * header. These will also be used to fill placeholders.\n   * @param {Object} [placeholders={}] The placeholders to fill in the content.\n   * These will not be included in the response header.\n   * @returns {Object} The Promise Router response.\n   */\n  async pageResponse(path, params = {}, placeholders = {}) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    // Get config placeholders; can be an object, a function or an async function\n    let configPlaceholders =\n      typeof this.pagesConfig.placeholders === 'function'\n        ? this.pagesConfig.placeholders(params)\n        : Object.prototype.toString.call(this.pagesConfig.placeholders) === '[object Object]'\n          ? this.pagesConfig.placeholders\n          : {};\n    if (configPlaceholders instanceof Promise) {\n      configPlaceholders = await configPlaceholders;\n    }\n\n    // Fill placeholders\n    const allPlaceholders = Object.assign({}, configPlaceholders, placeholders);\n    const paramsAndPlaceholders = Object.assign({}, params, allPlaceholders);\n    data = mustache.render(data, paramsAndPlaceholders);\n\n    // Add placeholders in header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return { text: data, headers: headers };\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @returns {Object} The PromiseRouter response.\n   */\n  async fileResponse(path) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    return { text: data };\n  }\n\n  /**\n   * Reads and returns the content of a file at a given path. File reading to\n   * serve content on the static route is only allowed from the pages\n   * directory on downwards.\n   * -----------------------------------------------------------------------\n   * **WARNING:** All file reads in the PagesRouter must be executed by this\n   * wrapper because it also detects and prevents common exploits.\n   * -----------------------------------------------------------------------\n   * @param {String} filePath The path to the file to read.\n   * @returns {Promise<String>} The file content.\n   */\n  async readFile(filePath) {\n    // Normalize path to prevent it from containing any directory changing\n    // UNIX patterns which could expose the whole file system, e.g.\n    // `http://example.com/parse/apps/../file.txt` requests a file outside\n    // of the pages directory scope.\n    const normalizedPath = path.normalize(filePath);\n\n    // Abort if the path is outside of the path directory scope\n    if (!normalizedPath.startsWith(this.pagesPath)) {\n      throw errors.fileOutsideAllowedScope;\n    }\n\n    return await fs.readFile(normalizedPath, 'utf-8');\n  }\n\n  /**\n   * Loads a language resource JSON file that is used for translations.\n   */\n  loadJsonResource() {\n    if (this.pagesConfig.localizationJsonPath === undefined) {\n      return;\n    }\n    try {\n      const json = require(path.resolve('./', this.pagesConfig.localizationJsonPath));\n      this.jsonParameters = json;\n    } catch (e) {\n      throw errors.jsonFailedFileLoading;\n    }\n  }\n\n  /**\n   * Extracts and returns the page default parameters from the Parse Server\n   * configuration. These parameters are made accessible in every page served\n   * by this router.\n   * @param {Object} config The Parse Server configuration.\n   * @returns {Object} The default parameters.\n   */\n  getDefaultParams(config) {\n    return config\n      ? {\n        [pageParams.appId]: config.appId,\n        [pageParams.appName]: config.appName,\n        [pageParams.publicServerUrl]: config.publicServerURL,\n      }\n      : {};\n  }\n\n  /**\n   * Extracts and returns the locale from an express request.\n   * @param {Object} req The express request.\n   * @returns {String|undefined} The locale, or undefined if no locale was set.\n   */\n  getLocale(req) {\n    const locale =\n      (req.query || {})[pageParams.locale] ||\n      (req.body || {})[pageParams.locale] ||\n      (req.params || {})[pageParams.locale] ||\n      (req.headers || {})[pageParamHeaderPrefix + pageParams.locale];\n    return locale;\n  }\n\n  /**\n   * Creates a response with http redirect.\n   * @param {Object} req The express request.\n   * @param {String} path The path of the file to return.\n   * @param {Object} params The query parameters to include.\n   * @returns {Object} The Promise Router response.\n   */\n  async redirectResponse(url, params) {\n    // Remove any parameters with undefined value\n    params = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[p[0]] = p[1];\n      }\n      return m;\n    }, {});\n\n    // Compose URL with parameters in query\n    const location = new URL(url);\n    Object.entries(params).forEach(p => location.searchParams.set(p[0], p[1]));\n    const locationString = location.toString();\n\n    // Add parameters to header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return {\n      status: 303,\n      location: locationString,\n      headers: headers,\n    };\n  }\n\n  defaultPagePath(file) {\n    return path.join(this.pagesPath, file);\n  }\n\n  composePageUrl(file, publicServerUrl, locale) {\n    let url = publicServerUrl;\n    url += url.endsWith('/') ? '' : '/';\n    url += this.pagesEndpoint + '/';\n    url += locale === undefined ? '' : locale + '/';\n    url += file;\n    return url;\n  }\n\n  notFound() {\n    return {\n      text: 'Not found.',\n      status: 404,\n    };\n  }\n\n  invalidRequest() {\n    const error = new Error();\n    error.status = 403;\n    error.message = 'unauthorized';\n    throw error;\n  }\n\n  /**\n   * Sets the Parse Server configuration in the request object to make it\n   * easily accessible throughtout request processing.\n   * @param {Object} req The request.\n   * @param {Boolean} failGracefully Is true if failing to set the config should\n   * not result in an invalid request response. Default is `false`.\n   */\n  setConfig(req, failGracefully = false) {\n    req.config = Config.get(req.params.appId || req.query.appId);\n    if (!req.config && !failGracefully) {\n      this.invalidRequest();\n    }\n    return Promise.resolve();\n  }\n\n  mountPagesRoutes() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/verify_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.verifyEmail(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/resend_verification_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resendVerificationEmail(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/choose_password`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.passwordReset(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resetPassword(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.requestResetPassword(req);\n      }\n    );\n  }\n\n  mountCustomRoutes() {\n    for (const route of this.pagesConfig.customRoutes || []) {\n      this.route(\n        route.method,\n        `/${this.pagesEndpoint}/:appId/${route.path}`,\n        req => {\n          this.setConfig(req);\n        },\n        async req => {\n          const { file, query = {} } = (await route.handler(req)) || {};\n\n          // If route handler did not return a page send 404 response\n          if (!file) {\n            return this.notFound();\n          }\n\n          // Send page response\n          const page = new Page({ id: file, defaultFile: file });\n          return this.goToPage(req, page, query, false);\n        }\n      );\n    }\n  }\n\n  mountStaticRoute() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/(*)?`,\n      req => {\n        this.setConfig(req, true);\n      },\n      req => {\n        return this.staticRoute(req);\n      }\n    );\n  }\n\n  expressRouter() {\n    const router = express.Router();\n    router.use('/', super.expressRouter());\n    return router;\n  }\n}\n\nexport default PagesRouter;\nmodule.exports = {\n  PagesRouter,\n  pageParamHeaderPrefix,\n  pageParams,\n  pages,\n};\n"],"mappings":"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"}