parse-server 9.2.0 → 9.2.1-alpha.1

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.
Files changed (30) hide show
  1. package/lib/Controllers/PushController.js +2 -2
  2. package/lib/Routers/PagesRouter.js +12 -10
  3. package/package.json +3 -3
  4. package/public/custom_json.html +17 -0
  5. package/public/custom_json.json +23 -0
  6. package/public/custom_page.html +15 -0
  7. package/public/de/email_verification_link_expired.html +24 -0
  8. package/public/de/email_verification_link_invalid.html +21 -0
  9. package/public/de/email_verification_send_fail.html +21 -0
  10. package/public/de/email_verification_send_success.html +19 -0
  11. package/public/de/email_verification_success.html +18 -0
  12. package/public/de/password_reset.html +65 -0
  13. package/public/de/password_reset_link_invalid.html +19 -0
  14. package/public/de/password_reset_success.html +18 -0
  15. package/public/de-AT/email_verification_link_expired.html +24 -0
  16. package/public/de-AT/email_verification_link_invalid.html +21 -0
  17. package/public/de-AT/email_verification_send_fail.html +21 -0
  18. package/public/de-AT/email_verification_send_success.html +19 -0
  19. package/public/de-AT/email_verification_success.html +18 -0
  20. package/public/de-AT/password_reset.html +65 -0
  21. package/public/de-AT/password_reset_link_invalid.html +19 -0
  22. package/public/de-AT/password_reset_success.html +18 -0
  23. package/public/email_verification_link_expired.html +24 -0
  24. package/public/email_verification_link_invalid.html +21 -0
  25. package/public/email_verification_send_fail.html +21 -0
  26. package/public/email_verification_send_success.html +19 -0
  27. package/public/email_verification_success.html +18 -0
  28. package/public/password_reset.html +65 -0
  29. package/public/password_reset_link_invalid.html +19 -0
  30. package/public/password_reset_success.html +18 -0
@@ -198,7 +198,7 @@ class PushController {
198
198
  static pushTimeHasTimezoneComponent(pushTimeParam) {
199
199
  const offsetPattern = /(.+)([+-])\d\d:\d\d$/;
200
200
  return pushTimeParam.indexOf('Z') === pushTimeParam.length - 1 || offsetPattern.test(pushTimeParam) // 2007-04-05T12:30Z
201
- ; // 2007-04-05T12:30.000+02:00, 2007-04-05T12:30.000-02:00
201
+ ; // 2007-04-05T12:30.000+02:00, 2007-04-05T12:30.000-02:00
202
202
  }
203
203
 
204
204
  /**
@@ -221,4 +221,4 @@ class PushController {
221
221
  }
222
222
  exports.PushController = PushController;
223
223
  var _default = exports.default = PushController;
224
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_node","require","_RestQuery","_interopRequireDefault","_RestWrite","_Auth","_StatusHandler","_utils","e","__esModule","default","PushController","sendPush","body","where","config","auth","onPushStatusSaved","now","Date","hasPushSupport","Parse","Error","PUSH_MISCONFIGURED","expiration_time","getExpirationTime","expiration_interval","getExpirationInterval","Object","prototype","hasOwnProperty","call","ttlMs","valueOf","pushTime","getPushTime","date","formatPushTime","badgeUpdate","Promise","resolve","data","badge","restUpdate","toLowerCase","__op","amount","Number","updateWhere","applyDeviceTokenExists","restQuery","RestQuery","method","Method","find","runBeforeFind","master","className","restWhere","buildRestWhere","then","write","RestWrite","runOptions","many","execute","pushStatus","pushStatusHandler","setInitial","objectId","audience_id","audienceId","updateAudience","lastUsed","__type","iso","toISOString","timesUsed","hasPushScheduledSupport","pushControllerQueue","enqueue","catch","err","fail","hasExpirationTime","expirationTimeParam","expirationTime","isFinite","hasExpirationInterval","expirationIntervalParam","hasPushTime","pushTimeParam","isLocalTime","pushTimeHasTimezoneComponent","offsetPattern","indexOf","length","test","isoString","substring","exports","_default"],"sources":["../../src/Controllers/PushController.js"],"sourcesContent":["import { Parse } from 'parse/node';\nimport RestQuery from '../RestQuery';\nimport RestWrite from '../RestWrite';\nimport { master } from '../Auth';\nimport { pushStatusHandler } from '../StatusHandler';\nimport { applyDeviceTokenExists } from '../Push/utils';\n\nexport class PushController {\n  sendPush(body = {}, where = {}, config, auth, onPushStatusSaved = () => {}, now = new Date()) {\n    if (!config.hasPushSupport) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED, 'Missing push configuration');\n    }\n\n    // Replace the expiration_time and push_time with a valid Unix epoch milliseconds time\n    body.expiration_time = PushController.getExpirationTime(body);\n    body.expiration_interval = PushController.getExpirationInterval(body);\n    if (body.expiration_time && body.expiration_interval) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        'Both expiration_time and expiration_interval cannot be set'\n      );\n    }\n\n    // Immediate push\n    if (body.expiration_interval && !Object.prototype.hasOwnProperty.call(body, 'push_time')) {\n      const ttlMs = body.expiration_interval * 1000;\n      body.expiration_time = new Date(now.valueOf() + ttlMs).valueOf();\n    }\n\n    const pushTime = PushController.getPushTime(body);\n    if (pushTime && pushTime.date !== 'undefined') {\n      body['push_time'] = PushController.formatPushTime(pushTime);\n    }\n\n    // TODO: If the req can pass the checking, we return immediately instead of waiting\n    // pushes to be sent. We probably change this behaviour in the future.\n    let badgeUpdate = () => {\n      return Promise.resolve();\n    };\n\n    if (body.data && body.data.badge) {\n      const badge = body.data.badge;\n      let restUpdate = {};\n      if (typeof badge == 'string' && badge.toLowerCase() === 'increment') {\n        restUpdate = { badge: { __op: 'Increment', amount: 1 } };\n      } else if (\n        typeof badge == 'object' &&\n        typeof badge.__op == 'string' &&\n        badge.__op.toLowerCase() == 'increment' &&\n        Number(badge.amount)\n      ) {\n        restUpdate = { badge: { __op: 'Increment', amount: badge.amount } };\n      } else if (Number(badge)) {\n        restUpdate = { badge: badge };\n      } else {\n        throw \"Invalid value for badge, expected number or 'Increment' or {increment: number}\";\n      }\n\n      // Force filtering on only valid device tokens\n      const updateWhere = applyDeviceTokenExists(where);\n      badgeUpdate = async () => {\n        // Build a real RestQuery so we can use it in RestWrite\n        const restQuery = await RestQuery({\n          method: RestQuery.Method.find,\n          config,\n          runBeforeFind: false,\n          auth: master(config),\n          className: '_Installation',\n          restWhere: updateWhere,\n        });\n        return restQuery.buildRestWhere().then(() => {\n          const write = new RestWrite(\n            config,\n            master(config),\n            '_Installation',\n            restQuery.restWhere,\n            restUpdate\n          );\n          write.runOptions.many = true;\n          return write.execute();\n        });\n      };\n    }\n    const pushStatus = pushStatusHandler(config);\n    return Promise.resolve()\n      .then(() => {\n        return pushStatus.setInitial(body, where);\n      })\n      .then(() => {\n        onPushStatusSaved(pushStatus.objectId);\n        return badgeUpdate();\n      })\n      .then(() => {\n        // Update audience lastUsed and timesUsed\n        if (body.audience_id) {\n          const audienceId = body.audience_id;\n\n          var updateAudience = {\n            lastUsed: { __type: 'Date', iso: new Date().toISOString() },\n            timesUsed: { __op: 'Increment', amount: 1 },\n          };\n          const write = new RestWrite(\n            config,\n            master(config),\n            '_Audience',\n            { objectId: audienceId },\n            updateAudience\n          );\n          write.execute();\n        }\n        // Don't wait for the audience update promise to resolve.\n        return Promise.resolve();\n      })\n      .then(() => {\n        if (\n          Object.prototype.hasOwnProperty.call(body, 'push_time') &&\n          config.hasPushScheduledSupport\n        ) {\n          return Promise.resolve();\n        }\n        return config.pushControllerQueue.enqueue(body, where, config, auth, pushStatus);\n      })\n      .catch(err => {\n        return pushStatus.fail(err).then(() => {\n          throw err;\n        });\n      });\n  }\n\n  /**\n   * Get expiration time from the request body.\n   * @param {Object} request A request object\n   * @returns {Number|undefined} The expiration time if it exists in the request\n   */\n  static getExpirationTime(body = {}) {\n    var hasExpirationTime = Object.prototype.hasOwnProperty.call(body, 'expiration_time');\n    if (!hasExpirationTime) {\n      return;\n    }\n    var expirationTimeParam = body['expiration_time'];\n    var expirationTime;\n    if (typeof expirationTimeParam === 'number') {\n      expirationTime = new Date(expirationTimeParam * 1000);\n    } else if (typeof expirationTimeParam === 'string') {\n      expirationTime = new Date(expirationTimeParam);\n    } else {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        body['expiration_time'] + ' is not valid time.'\n      );\n    }\n    // Check expirationTime is valid or not, if it is not valid, expirationTime is NaN\n    if (!isFinite(expirationTime)) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        body['expiration_time'] + ' is not valid time.'\n      );\n    }\n    return expirationTime.valueOf();\n  }\n\n  static getExpirationInterval(body = {}) {\n    const hasExpirationInterval = Object.prototype.hasOwnProperty.call(body, 'expiration_interval');\n    if (!hasExpirationInterval) {\n      return;\n    }\n\n    var expirationIntervalParam = body['expiration_interval'];\n    if (typeof expirationIntervalParam !== 'number' || expirationIntervalParam <= 0) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        `expiration_interval must be a number greater than 0`\n      );\n    }\n    return expirationIntervalParam;\n  }\n\n  /**\n   * Get push time from the request body.\n   * @param {Object} request A request object\n   * @returns {Number|undefined} The push time if it exists in the request\n   */\n  static getPushTime(body = {}) {\n    var hasPushTime = Object.prototype.hasOwnProperty.call(body, 'push_time');\n    if (!hasPushTime) {\n      return;\n    }\n    var pushTimeParam = body['push_time'];\n    var date;\n    var isLocalTime = true;\n\n    if (typeof pushTimeParam === 'number') {\n      date = new Date(pushTimeParam * 1000);\n    } else if (typeof pushTimeParam === 'string') {\n      isLocalTime = !PushController.pushTimeHasTimezoneComponent(pushTimeParam);\n      date = new Date(pushTimeParam);\n    } else {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        body['push_time'] + ' is not valid time.'\n      );\n    }\n    // Check pushTime is valid or not, if it is not valid, pushTime is NaN\n    if (!isFinite(date)) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        body['push_time'] + ' is not valid time.'\n      );\n    }\n\n    return {\n      date,\n      isLocalTime,\n    };\n  }\n\n  /**\n   * Checks if a ISO8601 formatted date contains a timezone component\n   * @param pushTimeParam {string}\n   * @returns {boolean}\n   */\n  static pushTimeHasTimezoneComponent(pushTimeParam: string): boolean {\n    const offsetPattern = /(.+)([+-])\\d\\d:\\d\\d$/;\n    return (\n      pushTimeParam.indexOf('Z') === pushTimeParam.length - 1 || offsetPattern.test(pushTimeParam) // 2007-04-05T12:30Z\n    ); // 2007-04-05T12:30.000+02:00, 2007-04-05T12:30.000-02:00\n  }\n\n  /**\n   * Converts a date to ISO format in UTC time and strips the timezone if `isLocalTime` is true\n   * @param date {Date}\n   * @param isLocalTime {boolean}\n   * @returns {string}\n   */\n  static formatPushTime({ date, isLocalTime }: { date: Date, isLocalTime: boolean }) {\n    if (isLocalTime) {\n      // Strip 'Z'\n      const isoString = date.toISOString();\n      return isoString.substring(0, isoString.indexOf('Z'));\n    }\n    return date.toISOString();\n  }\n}\n\nexport default PushController;\n"],"mappings":"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","ignoreList":[]}
224
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_node","require","_RestQuery","_interopRequireDefault","_RestWrite","_Auth","_StatusHandler","_utils","e","__esModule","default","PushController","sendPush","body","where","config","auth","onPushStatusSaved","now","Date","hasPushSupport","Parse","Error","PUSH_MISCONFIGURED","expiration_time","getExpirationTime","expiration_interval","getExpirationInterval","Object","prototype","hasOwnProperty","call","ttlMs","valueOf","pushTime","getPushTime","date","formatPushTime","badgeUpdate","Promise","resolve","data","badge","restUpdate","toLowerCase","__op","amount","Number","updateWhere","applyDeviceTokenExists","restQuery","RestQuery","method","Method","find","runBeforeFind","master","className","restWhere","buildRestWhere","then","write","RestWrite","runOptions","many","execute","pushStatus","pushStatusHandler","setInitial","objectId","audience_id","audienceId","updateAudience","lastUsed","__type","iso","toISOString","timesUsed","hasPushScheduledSupport","pushControllerQueue","enqueue","catch","err","fail","hasExpirationTime","expirationTimeParam","expirationTime","isFinite","hasExpirationInterval","expirationIntervalParam","hasPushTime","pushTimeParam","isLocalTime","pushTimeHasTimezoneComponent","offsetPattern","indexOf","length","test","isoString","substring","exports","_default"],"sources":["../../src/Controllers/PushController.js"],"sourcesContent":["import { Parse } from 'parse/node';\nimport RestQuery from '../RestQuery';\nimport RestWrite from '../RestWrite';\nimport { master } from '../Auth';\nimport { pushStatusHandler } from '../StatusHandler';\nimport { applyDeviceTokenExists } from '../Push/utils';\n\nexport class PushController {\n  sendPush(body = {}, where = {}, config, auth, onPushStatusSaved = () => {}, now = new Date()) {\n    if (!config.hasPushSupport) {\n      throw new Parse.Error(Parse.Error.PUSH_MISCONFIGURED, 'Missing push configuration');\n    }\n\n    // Replace the expiration_time and push_time with a valid Unix epoch milliseconds time\n    body.expiration_time = PushController.getExpirationTime(body);\n    body.expiration_interval = PushController.getExpirationInterval(body);\n    if (body.expiration_time && body.expiration_interval) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        'Both expiration_time and expiration_interval cannot be set'\n      );\n    }\n\n    // Immediate push\n    if (body.expiration_interval && !Object.prototype.hasOwnProperty.call(body, 'push_time')) {\n      const ttlMs = body.expiration_interval * 1000;\n      body.expiration_time = new Date(now.valueOf() + ttlMs).valueOf();\n    }\n\n    const pushTime = PushController.getPushTime(body);\n    if (pushTime && pushTime.date !== 'undefined') {\n      body['push_time'] = PushController.formatPushTime(pushTime);\n    }\n\n    // TODO: If the req can pass the checking, we return immediately instead of waiting\n    // pushes to be sent. We probably change this behaviour in the future.\n    let badgeUpdate = () => {\n      return Promise.resolve();\n    };\n\n    if (body.data && body.data.badge) {\n      const badge = body.data.badge;\n      let restUpdate = {};\n      if (typeof badge == 'string' && badge.toLowerCase() === 'increment') {\n        restUpdate = { badge: { __op: 'Increment', amount: 1 } };\n      } else if (\n        typeof badge == 'object' &&\n        typeof badge.__op == 'string' &&\n        badge.__op.toLowerCase() == 'increment' &&\n        Number(badge.amount)\n      ) {\n        restUpdate = { badge: { __op: 'Increment', amount: badge.amount } };\n      } else if (Number(badge)) {\n        restUpdate = { badge: badge };\n      } else {\n        throw \"Invalid value for badge, expected number or 'Increment' or {increment: number}\";\n      }\n\n      // Force filtering on only valid device tokens\n      const updateWhere = applyDeviceTokenExists(where);\n      badgeUpdate = async () => {\n        // Build a real RestQuery so we can use it in RestWrite\n        const restQuery = await RestQuery({\n          method: RestQuery.Method.find,\n          config,\n          runBeforeFind: false,\n          auth: master(config),\n          className: '_Installation',\n          restWhere: updateWhere,\n        });\n        return restQuery.buildRestWhere().then(() => {\n          const write = new RestWrite(\n            config,\n            master(config),\n            '_Installation',\n            restQuery.restWhere,\n            restUpdate\n          );\n          write.runOptions.many = true;\n          return write.execute();\n        });\n      };\n    }\n    const pushStatus = pushStatusHandler(config);\n    return Promise.resolve()\n      .then(() => {\n        return pushStatus.setInitial(body, where);\n      })\n      .then(() => {\n        onPushStatusSaved(pushStatus.objectId);\n        return badgeUpdate();\n      })\n      .then(() => {\n        // Update audience lastUsed and timesUsed\n        if (body.audience_id) {\n          const audienceId = body.audience_id;\n\n          var updateAudience = {\n            lastUsed: { __type: 'Date', iso: new Date().toISOString() },\n            timesUsed: { __op: 'Increment', amount: 1 },\n          };\n          const write = new RestWrite(\n            config,\n            master(config),\n            '_Audience',\n            { objectId: audienceId },\n            updateAudience\n          );\n          write.execute();\n        }\n        // Don't wait for the audience update promise to resolve.\n        return Promise.resolve();\n      })\n      .then(() => {\n        if (\n          Object.prototype.hasOwnProperty.call(body, 'push_time') &&\n          config.hasPushScheduledSupport\n        ) {\n          return Promise.resolve();\n        }\n        return config.pushControllerQueue.enqueue(body, where, config, auth, pushStatus);\n      })\n      .catch(err => {\n        return pushStatus.fail(err).then(() => {\n          throw err;\n        });\n      });\n  }\n\n  /**\n   * Get expiration time from the request body.\n   * @param {Object} request A request object\n   * @returns {Number|undefined} The expiration time if it exists in the request\n   */\n  static getExpirationTime(body = {}) {\n    var hasExpirationTime = Object.prototype.hasOwnProperty.call(body, 'expiration_time');\n    if (!hasExpirationTime) {\n      return;\n    }\n    var expirationTimeParam = body['expiration_time'];\n    var expirationTime;\n    if (typeof expirationTimeParam === 'number') {\n      expirationTime = new Date(expirationTimeParam * 1000);\n    } else if (typeof expirationTimeParam === 'string') {\n      expirationTime = new Date(expirationTimeParam);\n    } else {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        body['expiration_time'] + ' is not valid time.'\n      );\n    }\n    // Check expirationTime is valid or not, if it is not valid, expirationTime is NaN\n    if (!isFinite(expirationTime)) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        body['expiration_time'] + ' is not valid time.'\n      );\n    }\n    return expirationTime.valueOf();\n  }\n\n  static getExpirationInterval(body = {}) {\n    const hasExpirationInterval = Object.prototype.hasOwnProperty.call(body, 'expiration_interval');\n    if (!hasExpirationInterval) {\n      return;\n    }\n\n    var expirationIntervalParam = body['expiration_interval'];\n    if (typeof expirationIntervalParam !== 'number' || expirationIntervalParam <= 0) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        `expiration_interval must be a number greater than 0`\n      );\n    }\n    return expirationIntervalParam;\n  }\n\n  /**\n   * Get push time from the request body.\n   * @param {Object} request A request object\n   * @returns {Number|undefined} The push time if it exists in the request\n   */\n  static getPushTime(body = {}) {\n    var hasPushTime = Object.prototype.hasOwnProperty.call(body, 'push_time');\n    if (!hasPushTime) {\n      return;\n    }\n    var pushTimeParam = body['push_time'];\n    var date;\n    var isLocalTime = true;\n\n    if (typeof pushTimeParam === 'number') {\n      date = new Date(pushTimeParam * 1000);\n    } else if (typeof pushTimeParam === 'string') {\n      isLocalTime = !PushController.pushTimeHasTimezoneComponent(pushTimeParam);\n      date = new Date(pushTimeParam);\n    } else {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        body['push_time'] + ' is not valid time.'\n      );\n    }\n    // Check pushTime is valid or not, if it is not valid, pushTime is NaN\n    if (!isFinite(date)) {\n      throw new Parse.Error(\n        Parse.Error.PUSH_MISCONFIGURED,\n        body['push_time'] + ' is not valid time.'\n      );\n    }\n\n    return {\n      date,\n      isLocalTime,\n    };\n  }\n\n  /**\n   * Checks if a ISO8601 formatted date contains a timezone component\n   * @param pushTimeParam {string}\n   * @returns {boolean}\n   */\n  static pushTimeHasTimezoneComponent(pushTimeParam: string): boolean {\n    const offsetPattern = /(.+)([+-])\\d\\d:\\d\\d$/;\n    return (\n      pushTimeParam.indexOf('Z') === pushTimeParam.length - 1 || offsetPattern.test(pushTimeParam) // 2007-04-05T12:30Z\n    ); // 2007-04-05T12:30.000+02:00, 2007-04-05T12:30.000-02:00\n  }\n\n  /**\n   * Converts a date to ISO format in UTC time and strips the timezone if `isLocalTime` is true\n   * @param date {Date}\n   * @param isLocalTime {boolean}\n   * @returns {string}\n   */\n  static formatPushTime({ date, isLocalTime }: { date: Date, isLocalTime: boolean }) {\n    if (isLocalTime) {\n      // Strip 'Z'\n      const isoString = date.toISOString();\n      return isoString.substring(0, isoString.indexOf('Z'));\n    }\n    return date.toISOString();\n  }\n}\n\nexport default PushController;\n"],"mappings":"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","ignoreList":[]}
@@ -565,36 +565,38 @@ class PagesRouter extends _PromiseRouter.default {
565
565
  * @param {Boolean} failGracefully Is true if failing to set the config should
566
566
  * not result in an invalid request response. Default is `false`.
567
567
  */
568
- setConfig(req, failGracefully = false) {
568
+ async setConfig(req, failGracefully = false) {
569
569
  req.config = _Config.default.get(req.params.appId || req.query.appId);
570
570
  if (!req.config && !failGracefully) {
571
571
  this.invalidRequest();
572
572
  }
573
- return Promise.resolve();
573
+ if (req.config) {
574
+ await req.config.loadKeys();
575
+ }
574
576
  }
575
577
  mountPagesRoutes() {
576
578
  this.route('GET', `/${this.pagesEndpoint}/:appId/verify_email`, req => {
577
- this.setConfig(req);
579
+ return this.setConfig(req);
578
580
  }, req => {
579
581
  return this.verifyEmail(req);
580
582
  });
581
583
  this.route('POST', `/${this.pagesEndpoint}/:appId/resend_verification_email`, req => {
582
- this.setConfig(req);
584
+ return this.setConfig(req);
583
585
  }, req => {
584
586
  return this.resendVerificationEmail(req);
585
587
  });
586
588
  this.route('GET', `/${this.pagesEndpoint}/choose_password`, req => {
587
- this.setConfig(req);
589
+ return this.setConfig(req);
588
590
  }, req => {
589
591
  return this.passwordReset(req);
590
592
  });
591
593
  this.route('POST', `/${this.pagesEndpoint}/:appId/request_password_reset`, req => {
592
- this.setConfig(req);
594
+ return this.setConfig(req);
593
595
  }, req => {
594
596
  return this.resetPassword(req);
595
597
  });
596
598
  this.route('GET', `/${this.pagesEndpoint}/:appId/request_password_reset`, req => {
597
- this.setConfig(req);
599
+ return this.setConfig(req);
598
600
  }, req => {
599
601
  return this.requestResetPassword(req);
600
602
  });
@@ -602,7 +604,7 @@ class PagesRouter extends _PromiseRouter.default {
602
604
  mountCustomRoutes() {
603
605
  for (const route of this.pagesConfig.customRoutes || []) {
604
606
  this.route(route.method, `/${this.pagesEndpoint}/:appId/${route.path}`, req => {
605
- this.setConfig(req);
607
+ return this.setConfig(req);
606
608
  }, async req => {
607
609
  const {
608
610
  file,
@@ -625,7 +627,7 @@ class PagesRouter extends _PromiseRouter.default {
625
627
  }
626
628
  mountStaticRoute() {
627
629
  this.route('GET', `/${this.pagesEndpoint}/*resource`, req => {
628
- this.setConfig(req, true);
630
+ return this.setConfig(req, true);
629
631
  }, req => {
630
632
  return this.staticRoute(req);
631
633
  });
@@ -644,4 +646,4 @@ module.exports = {
644
646
  pageParams,
645
647
  pages
646
648
  };
647
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_PromiseRouter","_interopRequireDefault","require","_Config","_express","_path","_fs","_node","_Utils","_mustache","_Page","e","__esModule","default","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","resendVerificationEmail","body","params","publicServerURL","requestResetPassword","checkResetTokenValidity","applicationId","resetPassword","new_password","xhr","Parse","Error","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","configPlaceholders","prototype","call","allPlaceholders","paramsAndPlaceholders","headers","entries","reduce","m","p","toLowerCase","text","filePath","normalizedPath","normalize","startsWith","fs","json","url","location","URL","forEach","searchParams","set","locationString","file","join","message","setConfig","failGracefully","Config","get","route","customRoutes","handler","expressRouter","router","express","Router","use","exports","_default","module"],"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 { 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) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n    return userController.verifyEmail(token).then(\n      () => {\n        return this.goToPage(req, pages.emailVerificationSuccess);\n      },\n      () => {\n        return this.goToPage(req, pages.emailVerificationLinkInvalid);\n      }\n    );\n  }\n\n  resendVerificationEmail(req) {\n    const config = req.config;\n    const username = req.body?.username;\n    const token = req.body?.token;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!username && !token) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n\n    return userController.resendVerificationEmail(username, req, token).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 { token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!token) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    return config.userController.checkResetTokenValidity(token).then(\n      () => {\n        const params = {\n          [pageParams.token]: token,\n          [pageParams.appId]: config.applicationId,\n          [pageParams.appName]: config.appName,\n        };\n        return this.goToPage(req, pages.passwordReset, params);\n      },\n      () => {\n        return this.goToPage(req, pages.passwordResetLinkInvalid);\n      }\n    );\n  }\n\n  resetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { new_password, token: rawToken } = req.body || {};\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if ((!token || !new_password) && req.xhr === false) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\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(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          : {\n            [pageParams.token]: token,\n            [pageParams.appId]: config.applicationId,\n            [pageParams.error]: result.err,\n            [pageParams.appName]: config.appName,\n          };\n\n        if (result?.err === 'The password reset link has expired') {\n          delete query[pageParams.token];\n          query[pageParams.token] = token;\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['resource'][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 {\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 {\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 {\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}/*resource`,\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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","ignoreList":[]}
649
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_PromiseRouter","_interopRequireDefault","require","_Config","_express","_path","_fs","_node","_Utils","_mustache","_Page","e","__esModule","default","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","resendVerificationEmail","body","params","publicServerURL","requestResetPassword","checkResetTokenValidity","applicationId","resetPassword","new_password","xhr","Parse","Error","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","configPlaceholders","prototype","call","allPlaceholders","paramsAndPlaceholders","headers","entries","reduce","m","p","toLowerCase","text","filePath","normalizedPath","normalize","startsWith","fs","json","url","location","URL","forEach","searchParams","set","locationString","file","join","message","setConfig","failGracefully","Config","get","loadKeys","route","customRoutes","handler","expressRouter","router","express","Router","use","exports","_default","module"],"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 { 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) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n    return userController.verifyEmail(token).then(\n      () => {\n        return this.goToPage(req, pages.emailVerificationSuccess);\n      },\n      () => {\n        return this.goToPage(req, pages.emailVerificationLinkInvalid);\n      }\n    );\n  }\n\n  resendVerificationEmail(req) {\n    const config = req.config;\n    const username = req.body?.username;\n    const token = req.body?.token;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!username && !token) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n\n    return userController.resendVerificationEmail(username, req, token).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 { token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!token) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    return config.userController.checkResetTokenValidity(token).then(\n      () => {\n        const params = {\n          [pageParams.token]: token,\n          [pageParams.appId]: config.applicationId,\n          [pageParams.appName]: config.appName,\n        };\n        return this.goToPage(req, pages.passwordReset, params);\n      },\n      () => {\n        return this.goToPage(req, pages.passwordResetLinkInvalid);\n      }\n    );\n  }\n\n  resetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { new_password, token: rawToken } = req.body || {};\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if ((!token || !new_password) && req.xhr === false) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\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(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          : {\n            [pageParams.token]: token,\n            [pageParams.appId]: config.applicationId,\n            [pageParams.error]: result.err,\n            [pageParams.appName]: config.appName,\n          };\n\n        if (result?.err === 'The password reset link has expired') {\n          delete query[pageParams.token];\n          query[pageParams.token] = token;\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['resource'][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 {\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 {\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 {\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  async 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    if (req.config) {\n      await req.config.loadKeys();\n    }\n  }\n\n  mountPagesRoutes() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/verify_email`,\n      req => {\n        return 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        return 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        return 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        return 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        return 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          return 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}/*resource`,\n      req => {\n        return 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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","ignoreList":[]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "parse-server",
3
- "version": "9.2.0",
3
+ "version": "9.2.1-alpha.1",
4
4
  "description": "An express module providing a Parse-compatible API server",
5
5
  "main": "lib/index.js",
6
6
  "repository": {
@@ -10,7 +10,7 @@
10
10
  "files": [
11
11
  "bin/",
12
12
  "lib/",
13
- "public_html/",
13
+ "public/",
14
14
  "views/",
15
15
  "LICENSE",
16
16
  "NOTICE",
@@ -68,7 +68,7 @@
68
68
  "@actions/core": "1.11.1",
69
69
  "@apollo/client": "3.13.8",
70
70
  "@babel/cli": "7.27.0",
71
- "@babel/core": "7.28.6",
71
+ "@babel/core": "7.29.0",
72
72
  "@babel/eslint-parser": "7.28.6",
73
73
  "@babel/plugin-proposal-object-rest-spread": "7.20.7",
74
74
  "@babel/plugin-transform-flow-strip-types": "7.26.5",
@@ -0,0 +1,17 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page demonstrates how to localize using a JSON file that contains the
4
+ translations for each placeholder.
5
+ -->
6
+ <html>
7
+
8
+ <head>
9
+ <title>{{title}}</title>
10
+ </head>
11
+
12
+ <body>
13
+ <h1>{{heading}}</h1>
14
+ <p>{{body}}</p>
15
+ </body>
16
+
17
+ </html>
@@ -0,0 +1,23 @@
1
+ {
2
+ "en": {
3
+ "translation": {
4
+ "title": "Hello!",
5
+ "heading": "Welcome to {{appName}}!",
6
+ "body": "We are delighted to welcome you on board."
7
+ }
8
+ },
9
+ "de": {
10
+ "translation": {
11
+ "title": "Hallo!",
12
+ "heading": "Willkommen bei {{appName}}!",
13
+ "body": "Wir freuen uns, dich begrüßen zu dürfen."
14
+ }
15
+ },
16
+ "de-AT": {
17
+ "translation": {
18
+ "title": "Servus!",
19
+ "heading": "Willkommen bei {{appName}}!",
20
+ "body": "Wir freuen uns, dich begrüßen zu dürfen."
21
+ }
22
+ }
23
+ }
@@ -0,0 +1,15 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page demonstrates how to use custom pages in custom routes.
4
+ -->
5
+ <html>
6
+
7
+ <head>
8
+ <title>{{appName}}</title>
9
+ </head>
10
+
11
+ <body>
12
+ <h1>{{appName}}</h1>
13
+ </body>
14
+
15
+ </html>
@@ -0,0 +1,24 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link with a security
4
+ token that is expired or incorrect. This can either mean the user has clicked
5
+ on a stale link (i.e. re-clicked on the link) or this could be a sign of a
6
+ malicious user trying to tamper with your app.
7
+ -->
8
+ <html>
9
+
10
+ <head>
11
+ <title>Email Verification</title>
12
+ </head>
13
+
14
+ <body>
15
+ <h1>{{appName}}</h1>
16
+ <h1>Expired verification link!</h1>
17
+ <form method="POST" action="{{publicServerUrl}}/apps/{{appId}}/resend_verification_email">
18
+ <input name="token" type="hidden" value="{{token}}">
19
+ <input name="locale" type="hidden" value="{{locale}}">
20
+ <button type="submit">Resend Link</button>
21
+ </form>
22
+ </body>
23
+
24
+ </html>
@@ -0,0 +1,21 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link with parameters
4
+ that are missing or incorrect. This can either mean the user has incorrectly
5
+ entered a link or this could be a sign of a malicious user trying to tamper
6
+ with your app.
7
+ If the link contains an expired security token (or the email has already
8
+ been verified), this page is not displayed, there is another page for that.
9
+ -->
10
+ <html>
11
+
12
+ <head>
13
+ <title>Email Verification</title>
14
+ </head>
15
+
16
+ <body>
17
+ <h1>{{appName}}</h1>
18
+ <h1>Invalid verification link!</h1>
19
+ </body>
20
+
21
+ </html>
@@ -0,0 +1,21 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link with a security
4
+ token that is expired or incorrect, then requests to receive another link,
5
+ but it fails because the username is invalid or the email has already been
6
+ verified. This can either mean the user has previously verified the email
7
+ or this could be a sign of a malicious user trying to tamper with your app.
8
+ -->
9
+ <html>
10
+
11
+ <head>
12
+ <title>Email Verification</title>
13
+ </head>
14
+
15
+ <body>
16
+ <h1>{{appName}}</h1>
17
+ <h1>Invalid link!</h1>
18
+ <p>No link sent. User not found or email already verified.</p>
19
+ </body>
20
+
21
+ </html>
@@ -0,0 +1,19 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link with a
4
+ security token that is expired, then requests to receive another email
5
+ with a new link and the email is sent successfully.
6
+ -->
7
+ <html>
8
+
9
+ <head>
10
+ <title>Email Verification</title>
11
+ </head>
12
+
13
+ <body>
14
+ <h1>{{appName}}</h1>
15
+ <h1>Link sent!</h1>
16
+ <p>A new link has been sent. Check your email.</p>
17
+ </body>
18
+
19
+ </html>
@@ -0,0 +1,18 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link and the
4
+ email gets verified successfully.
5
+ -->
6
+ <html>
7
+
8
+ <head>
9
+ <title>Email Verification</title>
10
+ </head>
11
+
12
+ <body>
13
+ <h1>{{appName}}</h1>
14
+ <h1>Email verified!</h1>
15
+ <p>Successfully verified your email for account: {{username}}.</p>
16
+ </body>
17
+
18
+ </html>
@@ -0,0 +1,65 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when someone clicks a valid 'reset password' link.
4
+ Users should feel free to add to this page (i.e. branding or security widgets)
5
+ but should be sure not to delete any of the form inputs or the javascript from the
6
+ template file. This javascript is what adds the necessary values to authenticate
7
+ this session with Parse.
8
+ The query params 'username' and 'app' hold the friendly names for your current user and
9
+ your app. You should feel free to incorporate their values to make the page more personal.
10
+ If you are missing form parameters in your POST, Parse will navigate back to this page and
11
+ add an 'error' query parameter.
12
+ -->
13
+ <html>
14
+
15
+ <head>
16
+ <title>Password Reset</title>
17
+ </head>
18
+
19
+ <body>
20
+ <h1>{{appName}}</h1>
21
+ <h1>Reset Your Password</h1>
22
+ <noscript><p>We apologize, but resetting your password requires javascript</p></noscript>
23
+ <p>You can set a new Password for your account: {{username}}</p>
24
+ <br />
25
+ <p>{{error}}</p>
26
+ <form id='form' action='{{publicServerUrl}}/apps/{{appId}}/request_password_reset' method='POST'>
27
+ <input name='utf-8' type='hidden' value='✓' />
28
+ <input name="username" type="hidden" id="username" value="{{username}}" />
29
+ <input name="token" type="hidden" id="token" value="{{token}}" />
30
+ <input name="locale" type="hidden" id="locale" value="{{locale}}" />
31
+
32
+ <p>New Password</p>
33
+ <input name="new_password" type="password" id="password" />
34
+ <p>Confirm New Password</p>
35
+ <input name="confirm_new_password" type="password" id="password_confirm" />
36
+ <br />
37
+ <p id="password_match_info"></p>
38
+ <br />
39
+ <button id="change_password">Change Password</button>
40
+ </form>
41
+
42
+ <script>
43
+ window.onload = function() {
44
+ document.getElementById("password").oninput = validatePassword;
45
+ document.getElementById("password_confirm").oninput = validatePassword;
46
+ document.getElementById("change_password").disabled = true;
47
+
48
+ function validatePassword() {
49
+ var pass2 = document.getElementById("password").value;
50
+ var pass1 = document.getElementById("password_confirm").value;
51
+ if(pass1 !== pass2) {
52
+ if(document.getElementById("password_confirm").value) {
53
+ document.getElementById("change_password").disabled = true;
54
+ document.getElementById("password_match_info").innerHTML = "Must match the previous entry";
55
+ }
56
+ } else {
57
+ document.getElementById("change_password").disabled = false;
58
+ document.getElementById("password_match_info").innerHTML = "";
59
+ }
60
+ }
61
+ }
62
+ </script>
63
+ </body>
64
+
65
+ </html>
@@ -0,0 +1,19 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a password reset link with parameters
4
+ that are missing or incorrect. This can either mean the user has incorrectly
5
+ entered a link or this could be a sign of a malicious user trying to tamper
6
+ with your app.
7
+ -->
8
+ <html>
9
+
10
+ <head>
11
+ <title>Password Reset</title>
12
+ </head>
13
+
14
+ <body>
15
+ <h1>{{appName}}</h1>
16
+ <h1>Invalid password reset link!</h1>
17
+ </body>
18
+
19
+ </html>
@@ -0,0 +1,18 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a reset password link, then submits
4
+ the form with a new password and the password gets updated successfully.
5
+ -->
6
+ <html>
7
+
8
+ <head>
9
+ <title>Password Reset</title>
10
+ </head>
11
+
12
+ <body>
13
+ <h1>{{appName}}</h1>
14
+ <h1>Success!</h1>
15
+ <p>Your password has been updated.</p>
16
+ </body>
17
+
18
+ </html>
@@ -0,0 +1,24 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link with a security
4
+ token that is expired or incorrect. This can either mean the user has clicked
5
+ on a stale link (i.e. re-clicked on the link) or this could be a sign of a
6
+ malicious user trying to tamper with your app.
7
+ -->
8
+ <html>
9
+
10
+ <head>
11
+ <title>Email Verification</title>
12
+ </head>
13
+
14
+ <body>
15
+ <h1>{{appName}}</h1>
16
+ <h1>Expired verification link!</h1>
17
+ <form method="POST" action="{{publicServerUrl}}/apps/{{appId}}/resend_verification_email">
18
+ <input name="token" type="hidden" value="{{token}}">
19
+ <input name="locale" type="hidden" value="{{locale}}">
20
+ <button type="submit">Resend Link</button>
21
+ </form>
22
+ </body>
23
+
24
+ </html>
@@ -0,0 +1,21 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link with parameters
4
+ that are missing or incorrect. This can either mean the user has incorrectly
5
+ entered a link or this could be a sign of a malicious user trying to tamper
6
+ with your app.
7
+ If the link contains an expired security token (or the email has already
8
+ been verified), this page is not displayed, there is another page for that.
9
+ -->
10
+ <html>
11
+
12
+ <head>
13
+ <title>Email Verification</title>
14
+ </head>
15
+
16
+ <body>
17
+ <h1>{{appName}}</h1>
18
+ <h1>Invalid verification link!</h1>
19
+ </body>
20
+
21
+ </html>
@@ -0,0 +1,21 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link with a security
4
+ token that is expired or incorrect, then requests to receive another link,
5
+ but it fails because the username is invalid or the email has already been
6
+ verified. This can either mean the user has previously verified the email
7
+ or this could be a sign of a malicious user trying to tamper with your app.
8
+ -->
9
+ <html>
10
+
11
+ <head>
12
+ <title>Email Verification</title>
13
+ </head>
14
+
15
+ <body>
16
+ <h1>{{appName}}</h1>
17
+ <h1>Invalid link!</h1>
18
+ <p>No link sent. User not found or email already verified.</p>
19
+ </body>
20
+
21
+ </html>
@@ -0,0 +1,19 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link with a
4
+ security token that is expired, then requests to receive another email
5
+ with a new link and the email is sent successfully.
6
+ -->
7
+ <html>
8
+
9
+ <head>
10
+ <title>Email Verification</title>
11
+ </head>
12
+
13
+ <body>
14
+ <h1>{{appName}}</h1>
15
+ <h1>Link sent!</h1>
16
+ <p>A new link has been sent. Check your email.</p>
17
+ </body>
18
+
19
+ </html>
@@ -0,0 +1,18 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link and the
4
+ email gets verified successfully.
5
+ -->
6
+ <html>
7
+
8
+ <head>
9
+ <title>Email Verification</title>
10
+ </head>
11
+
12
+ <body>
13
+ <h1>{{appName}}</h1>
14
+ <h1>Email verified!</h1>
15
+ <p>Successfully verified your email for account: {{username}}.</p>
16
+ </body>
17
+
18
+ </html>
@@ -0,0 +1,65 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when someone clicks a valid 'reset password' link.
4
+ Users should feel free to add to this page (i.e. branding or security widgets)
5
+ but should be sure not to delete any of the form inputs or the javascript from the
6
+ template file. This javascript is what adds the necessary values to authenticate
7
+ this session with Parse.
8
+ The query params 'username' and 'app' hold the friendly names for your current user and
9
+ your app. You should feel free to incorporate their values to make the page more personal.
10
+ If you are missing form parameters in your POST, Parse will navigate back to this page and
11
+ add an 'error' query parameter.
12
+ -->
13
+ <html>
14
+
15
+ <head>
16
+ <title>Password Reset</title>
17
+ </head>
18
+
19
+ <body>
20
+ <h1>{{appName}}</h1>
21
+ <h1>Reset Your Password</h1>
22
+ <noscript><p>We apologize, but resetting your password requires javascript</p></noscript>
23
+ <p>You can set a new Password for your account: {{username}}</p>
24
+ <br />
25
+ <p>{{error}}</p>
26
+ <form id='form' action='{{publicServerUrl}}/apps/{{appId}}/request_password_reset' method='POST'>
27
+ <input name='utf-8' type='hidden' value='✓' />
28
+ <input name="username" type="hidden" id="username" value="{{username}}" />
29
+ <input name="token" type="hidden" id="token" value="{{token}}" />
30
+ <input name="locale" type="hidden" id="locale" value="{{locale}}" />
31
+
32
+ <p>New Password</p>
33
+ <input name="new_password" type="password" id="password" />
34
+ <p>Confirm New Password</p>
35
+ <input name="confirm_new_password" type="password" id="password_confirm" />
36
+ <br />
37
+ <p id="password_match_info"></p>
38
+ <br />
39
+ <button id="change_password">Change Password</button>
40
+ </form>
41
+
42
+ <script>
43
+ window.onload = function() {
44
+ document.getElementById("password").oninput = validatePassword;
45
+ document.getElementById("password_confirm").oninput = validatePassword;
46
+ document.getElementById("change_password").disabled = true;
47
+
48
+ function validatePassword() {
49
+ var pass2 = document.getElementById("password").value;
50
+ var pass1 = document.getElementById("password_confirm").value;
51
+ if(pass1 !== pass2) {
52
+ if(document.getElementById("password_confirm").value) {
53
+ document.getElementById("change_password").disabled = true;
54
+ document.getElementById("password_match_info").innerHTML = "Must match the previous entry";
55
+ }
56
+ } else {
57
+ document.getElementById("change_password").disabled = false;
58
+ document.getElementById("password_match_info").innerHTML = "";
59
+ }
60
+ }
61
+ }
62
+ </script>
63
+ </body>
64
+
65
+ </html>
@@ -0,0 +1,19 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a password reset link with parameters
4
+ that are missing or incorrect. This can either mean the user has incorrectly
5
+ entered a link or this could be a sign of a malicious user trying to tamper
6
+ with your app.
7
+ -->
8
+ <html>
9
+
10
+ <head>
11
+ <title>Password Reset</title>
12
+ </head>
13
+
14
+ <body>
15
+ <h1>{{appName}}</h1>
16
+ <h1>Invalid password reset link!</h1>
17
+ </body>
18
+
19
+ </html>
@@ -0,0 +1,18 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a reset password link, then submits
4
+ the form with a new password and the password gets updated successfully.
5
+ -->
6
+ <html>
7
+
8
+ <head>
9
+ <title>Password Reset</title>
10
+ </head>
11
+
12
+ <body>
13
+ <h1>{{appName}}</h1>
14
+ <h1>Success!</h1>
15
+ <p>Your password has been updated.</p>
16
+ </body>
17
+
18
+ </html>
@@ -0,0 +1,24 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link with a security
4
+ token that is expired or incorrect. This can either mean the user has clicked
5
+ on a stale link (i.e. re-clicked on the link) or this could be a sign of a
6
+ malicious user trying to tamper with your app.
7
+ -->
8
+ <html>
9
+
10
+ <head>
11
+ <title>Email Verification</title>
12
+ </head>
13
+
14
+ <body>
15
+ <h1>{{appName}}</h1>
16
+ <h1>Expired verification link!</h1>
17
+ <form method="POST" action="{{publicServerUrl}}/apps/{{appId}}/resend_verification_email">
18
+ <input name="token" type="hidden" value="{{token}}">
19
+ <input name="locale" type="hidden" value="{{locale}}">
20
+ <button type="submit">Resend Link</button>
21
+ </form>
22
+ </body>
23
+
24
+ </html>
@@ -0,0 +1,21 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link with parameters
4
+ that are missing or incorrect. This can either mean the user has incorrectly
5
+ entered a link or this could be a sign of a malicious user trying to tamper
6
+ with your app.
7
+ If the link contains an expired security token (or the email has already
8
+ been verified), this page is not displayed, there is another page for that.
9
+ -->
10
+ <html>
11
+
12
+ <head>
13
+ <title>Email Verification</title>
14
+ </head>
15
+
16
+ <body>
17
+ <h1>{{appName}}</h1>
18
+ <h1>Invalid verification link!</h1>
19
+ </body>
20
+
21
+ </html>
@@ -0,0 +1,21 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link with a security
4
+ token that is expired or incorrect, then requests to receive another link,
5
+ but it fails because the username is invalid or the email has already been
6
+ verified. This can either mean the user has previously verified the email
7
+ or this could be a sign of a malicious user trying to tamper with your app.
8
+ -->
9
+ <html>
10
+
11
+ <head>
12
+ <title>Email Verification</title>
13
+ </head>
14
+
15
+ <body>
16
+ <h1>{{appName}}</h1>
17
+ <h1>Invalid link!</h1>
18
+ <p>No link sent. User not found or email already verified.</p>
19
+ </body>
20
+
21
+ </html>
@@ -0,0 +1,19 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link with a
4
+ security token that is expired, then requests to receive another email
5
+ with a new link and the email is sent successfully.
6
+ -->
7
+ <html>
8
+
9
+ <head>
10
+ <title>Email Verification</title>
11
+ </head>
12
+
13
+ <body>
14
+ <h1>{{appName}}</h1>
15
+ <h1>Link sent!</h1>
16
+ <p>A new link has been sent. Check your email.</p>
17
+ </body>
18
+
19
+ </html>
@@ -0,0 +1,18 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a verify email link and the
4
+ email gets verified successfully.
5
+ -->
6
+ <html>
7
+
8
+ <head>
9
+ <title>Email Verification</title>
10
+ </head>
11
+
12
+ <body>
13
+ <h1>{{appName}}</h1>
14
+ <h1>Email verified!</h1>
15
+ <p>Successfully verified your email for account: {{username}}.</p>
16
+ </body>
17
+
18
+ </html>
@@ -0,0 +1,65 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when someone clicks a valid 'reset password' link.
4
+ Users should feel free to add to this page (i.e. branding or security widgets)
5
+ but should be sure not to delete any of the form inputs or the javascript from the
6
+ template file. This javascript is what adds the necessary values to authenticate
7
+ this session with Parse.
8
+ The query params 'username' and 'app' hold the friendly names for your current user and
9
+ your app. You should feel free to incorporate their values to make the page more personal.
10
+ If you are missing form parameters in your POST, Parse will navigate back to this page and
11
+ add an 'error' query parameter.
12
+ -->
13
+ <html>
14
+
15
+ <head>
16
+ <title>Password Reset</title>
17
+ </head>
18
+
19
+ <body>
20
+ <h1>{{appName}}</h1>
21
+ <h1>Reset Your Password</h1>
22
+ <noscript><p>We apologize, but resetting your password requires javascript</p></noscript>
23
+ <p>You can set a new Password for your account: {{username}}</p>
24
+ <br />
25
+ <p>{{error}}</p>
26
+ <form id='form' action='{{publicServerUrl}}/apps/{{appId}}/request_password_reset' method='POST'>
27
+ <input name='utf-8' type='hidden' value='✓' />
28
+ <input name="username" type="hidden" id="username" value="{{username}}" />
29
+ <input name="token" type="hidden" id="token" value="{{token}}" />
30
+ <input name="locale" type="hidden" id="locale" value="{{locale}}" />
31
+
32
+ <p>New Password</p>
33
+ <input name="new_password" type="password" id="password" />
34
+ <p>Confirm New Password</p>
35
+ <input name="confirm_new_password" type="password" id="password_confirm" />
36
+ <br />
37
+ <p id="password_match_info"></p>
38
+ <br />
39
+ <button id="change_password">Change Password</button>
40
+ </form>
41
+
42
+ <script>
43
+ window.onload = function() {
44
+ document.getElementById("password").oninput = validatePassword;
45
+ document.getElementById("password_confirm").oninput = validatePassword;
46
+ document.getElementById("change_password").disabled = true;
47
+
48
+ function validatePassword() {
49
+ var pass2 = document.getElementById("password").value;
50
+ var pass1 = document.getElementById("password_confirm").value;
51
+ if(pass1 !== pass2) {
52
+ if(document.getElementById("password_confirm").value) {
53
+ document.getElementById("change_password").disabled = true;
54
+ document.getElementById("password_match_info").innerHTML = "Must match the previous entry";
55
+ }
56
+ } else {
57
+ document.getElementById("change_password").disabled = false;
58
+ document.getElementById("password_match_info").innerHTML = "";
59
+ }
60
+ }
61
+ }
62
+ </script>
63
+ </body>
64
+
65
+ </html>
@@ -0,0 +1,19 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a password reset link with parameters
4
+ that are missing or incorrect. This can either mean the user has incorrectly
5
+ entered a link or this could be a sign of a malicious user trying to tamper
6
+ with your app.
7
+ -->
8
+ <html>
9
+
10
+ <head>
11
+ <title>Password Reset</title>
12
+ </head>
13
+
14
+ <body>
15
+ <h1>{{appName}}</h1>
16
+ <h1>Invalid password reset link!</h1>
17
+ </body>
18
+
19
+ </html>
@@ -0,0 +1,18 @@
1
+ <!DOCTYPE html>
2
+ <!--
3
+ This page is displayed when a user opens a reset password link, then submits
4
+ the form with a new password and the password gets updated successfully.
5
+ -->
6
+ <html>
7
+
8
+ <head>
9
+ <title>Password Reset</title>
10
+ </head>
11
+
12
+ <body>
13
+ <h1>{{appName}}</h1>
14
+ <h1>Success!</h1>
15
+ <p>Your password has been updated.</p>
16
+ </body>
17
+
18
+ </html>