123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661 |
- "use strict";
- Object.defineProperty(exports, "__esModule", {
- value: true
- });
- exports.default = exports.PagesRouter = void 0;
- var _PromiseRouter = _interopRequireDefault(require("../PromiseRouter"));
- var _Config = _interopRequireDefault(require("../Config"));
- var _express = _interopRequireDefault(require("express"));
- var _path = _interopRequireDefault(require("path"));
- var _fs = require("fs");
- var _node = require("parse/node");
- var _Utils = _interopRequireDefault(require("../Utils"));
- var _mustache = _interopRequireDefault(require("mustache"));
- var _Page = _interopRequireDefault(require("../Page"));
- function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
- // All pages with custom page key for reference and file name
- const pages = Object.freeze({
- passwordReset: new _Page.default({
- id: 'passwordReset',
- defaultFile: 'password_reset.html'
- }),
- passwordResetSuccess: new _Page.default({
- id: 'passwordResetSuccess',
- defaultFile: 'password_reset_success.html'
- }),
- passwordResetLinkInvalid: new _Page.default({
- id: 'passwordResetLinkInvalid',
- defaultFile: 'password_reset_link_invalid.html'
- }),
- emailVerificationSuccess: new _Page.default({
- id: 'emailVerificationSuccess',
- defaultFile: 'email_verification_success.html'
- }),
- emailVerificationSendFail: new _Page.default({
- id: 'emailVerificationSendFail',
- defaultFile: 'email_verification_send_fail.html'
- }),
- emailVerificationSendSuccess: new _Page.default({
- id: 'emailVerificationSendSuccess',
- defaultFile: 'email_verification_send_success.html'
- }),
- emailVerificationLinkInvalid: new _Page.default({
- id: 'emailVerificationLinkInvalid',
- defaultFile: 'email_verification_link_invalid.html'
- }),
- emailVerificationLinkExpired: new _Page.default({
- id: 'emailVerificationLinkExpired',
- defaultFile: 'email_verification_link_expired.html'
- })
- });
- // All page parameters for reference to be used as template placeholders or query params
- const pageParams = Object.freeze({
- appName: 'appName',
- appId: 'appId',
- token: 'token',
- username: 'username',
- error: 'error',
- locale: 'locale',
- publicServerUrl: 'publicServerUrl'
- });
- // The header prefix to add page params as response headers
- const pageParamHeaderPrefix = 'x-parse-page-param-';
- // The errors being thrown
- const errors = Object.freeze({
- jsonFailedFileLoading: 'failed to load JSON file',
- fileOutsideAllowedScope: 'not allowed to read file outside of pages directory'
- });
- class PagesRouter extends _PromiseRouter.default {
- /**
- * Constructs a PagesRouter.
- * @param {Object} pages The pages options from the Parse Server configuration.
- */
- constructor(pages = {}) {
- super();
- // Set instance properties
- this.pagesConfig = pages;
- this.pagesEndpoint = pages.pagesEndpoint ? pages.pagesEndpoint : 'apps';
- this.pagesPath = pages.pagesPath ? _path.default.resolve('./', pages.pagesPath) : _path.default.resolve(__dirname, '../../public');
- this.loadJsonResource();
- this.mountPagesRoutes();
- this.mountCustomRoutes();
- this.mountStaticRoute();
- }
- verifyEmail(req) {
- const config = req.config;
- const {
- username,
- token: rawToken
- } = req.query;
- const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;
- if (!config) {
- this.invalidRequest();
- }
- if (!token || !username) {
- return this.goToPage(req, pages.emailVerificationLinkInvalid);
- }
- const userController = config.userController;
- return userController.verifyEmail(username, token).then(() => {
- const params = {
- [pageParams.username]: username
- };
- return this.goToPage(req, pages.emailVerificationSuccess, params);
- }, () => {
- const params = {
- [pageParams.username]: username
- };
- return this.goToPage(req, pages.emailVerificationLinkExpired, params);
- });
- }
- resendVerificationEmail(req) {
- const config = req.config;
- const username = req.body.username;
- if (!config) {
- this.invalidRequest();
- }
- if (!username) {
- return this.goToPage(req, pages.emailVerificationLinkInvalid);
- }
- const userController = config.userController;
- return userController.resendVerificationEmail(username, req).then(() => {
- return this.goToPage(req, pages.emailVerificationSendSuccess);
- }, () => {
- return this.goToPage(req, pages.emailVerificationSendFail);
- });
- }
- passwordReset(req) {
- const config = req.config;
- const params = {
- [pageParams.appId]: req.params.appId,
- [pageParams.appName]: config.appName,
- [pageParams.token]: req.query.token,
- [pageParams.username]: req.query.username,
- [pageParams.publicServerUrl]: config.publicServerURL
- };
- return this.goToPage(req, pages.passwordReset, params);
- }
- requestResetPassword(req) {
- const config = req.config;
- if (!config) {
- this.invalidRequest();
- }
- const {
- username,
- token: rawToken
- } = req.query;
- const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;
- if (!username || !token) {
- return this.goToPage(req, pages.passwordResetLinkInvalid);
- }
- return config.userController.checkResetTokenValidity(username, token).then(() => {
- const params = {
- [pageParams.token]: token,
- [pageParams.username]: username,
- [pageParams.appId]: config.applicationId,
- [pageParams.appName]: config.appName
- };
- return this.goToPage(req, pages.passwordReset, params);
- }, () => {
- const params = {
- [pageParams.username]: username
- };
- return this.goToPage(req, pages.passwordResetLinkInvalid, params);
- });
- }
- resetPassword(req) {
- const config = req.config;
- if (!config) {
- this.invalidRequest();
- }
- const {
- username,
- new_password,
- token: rawToken
- } = req.body;
- const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;
- if ((!username || !token || !new_password) && req.xhr === false) {
- return this.goToPage(req, pages.passwordResetLinkInvalid);
- }
- if (!username) {
- throw new _node.Parse.Error(_node.Parse.Error.USERNAME_MISSING, 'Missing username');
- }
- if (!token) {
- throw new _node.Parse.Error(_node.Parse.Error.OTHER_CAUSE, 'Missing token');
- }
- if (!new_password) {
- throw new _node.Parse.Error(_node.Parse.Error.PASSWORD_MISSING, 'Missing password');
- }
- return config.userController.updatePassword(username, token, new_password).then(() => {
- return Promise.resolve({
- success: true
- });
- }, err => {
- return Promise.resolve({
- success: false,
- err
- });
- }).then(result => {
- if (req.xhr) {
- if (result.success) {
- return Promise.resolve({
- status: 200,
- response: 'Password successfully reset'
- });
- }
- if (result.err) {
- throw new _node.Parse.Error(_node.Parse.Error.OTHER_CAUSE, `${result.err}`);
- }
- }
- const query = result.success ? {
- [pageParams.username]: username
- } : {
- [pageParams.username]: username,
- [pageParams.token]: token,
- [pageParams.appId]: config.applicationId,
- [pageParams.error]: result.err,
- [pageParams.appName]: config.appName
- };
- const page = result.success ? pages.passwordResetSuccess : pages.passwordReset;
- return this.goToPage(req, page, query, false);
- });
- }
- /**
- * Returns page content if the page is a local file or returns a
- * redirect to a custom page.
- * @param {Object} req The express request.
- * @param {Page} page The page to go to.
- * @param {Object} [params={}] The query parameters to attach to the URL in case of
- * HTTP redirect responses for POST requests, or the placeholders to fill into
- * the response content in case of HTTP content responses for GET requests.
- * @param {Boolean} [responseType] Is true if a redirect response should be forced,
- * false if a content response should be forced, undefined if the response type
- * should depend on the request type by default:
- * - GET request -> content response
- * - POST request -> redirect response (PRG pattern)
- * @returns {Promise<Object>} The PromiseRouter response.
- */
- goToPage(req, page, params = {}, responseType) {
- const config = req.config;
- // Determine redirect either by force, response setting or request method
- const redirect = config.pages.forceRedirect ? true : responseType !== undefined ? responseType : req.method == 'POST';
- // Include default parameters
- const defaultParams = this.getDefaultParams(config);
- if (Object.values(defaultParams).includes(undefined)) {
- return this.notFound();
- }
- params = Object.assign(params, defaultParams);
- // Add locale to params to ensure it is passed on with every request;
- // that means, once a locale is set, it is passed on to any follow-up page,
- // e.g. request_password_reset -> password_reset -> password_reset_success
- const locale = this.getLocale(req);
- params[pageParams.locale] = locale;
- // Compose paths and URLs
- const defaultFile = page.defaultFile;
- const defaultPath = this.defaultPagePath(defaultFile);
- const defaultUrl = this.composePageUrl(defaultFile, config.publicServerURL);
- // If custom URL is set redirect to it without localization
- const customUrl = config.pages.customUrls[page.id];
- if (customUrl && !_Utils.default.isPath(customUrl)) {
- return this.redirectResponse(customUrl, params);
- }
- // Get JSON placeholders
- let placeholders = {};
- if (config.pages.enableLocalization && config.pages.localizationJsonPath) {
- placeholders = this.getJsonPlaceholders(locale, params);
- }
- // Send response
- if (config.pages.enableLocalization && locale) {
- return _Utils.default.getLocalizedPath(defaultPath, locale).then(({
- path,
- subdir
- }) => redirect ? this.redirectResponse(this.composePageUrl(defaultFile, config.publicServerURL, subdir), params) : this.pageResponse(path, params, placeholders));
- } else {
- return redirect ? this.redirectResponse(defaultUrl, params) : this.pageResponse(defaultPath, params, placeholders);
- }
- }
- /**
- * Serves a request to a static resource and localizes the resource if it
- * is a HTML file.
- * @param {Object} req The request object.
- * @returns {Promise<Object>} The response.
- */
- staticRoute(req) {
- // Get requested path
- const relativePath = req.params[0];
- // Resolve requested path to absolute path
- const absolutePath = _path.default.resolve(this.pagesPath, relativePath);
- // If the requested file is not a HTML file send its raw content
- if (!absolutePath || !absolutePath.endsWith('.html')) {
- return this.fileResponse(absolutePath);
- }
- // Get parameters
- const params = this.getDefaultParams(req.config);
- const locale = this.getLocale(req);
- if (locale) {
- params.locale = locale;
- }
- // Get JSON placeholders
- const placeholders = this.getJsonPlaceholders(locale, params);
- return this.pageResponse(absolutePath, params, placeholders);
- }
- /**
- * Returns a translation from the JSON resource for a given locale. The JSON
- * resource is parsed according to i18next syntax.
- *
- * Example JSON content:
- * ```js
- * {
- * "en": { // resource for language `en` (English)
- * "translation": {
- * "greeting": "Hello!"
- * }
- * },
- * "de": { // resource for language `de` (German)
- * "translation": {
- * "greeting": "Hallo!"
- * }
- * }
- * "de-CH": { // resource for locale `de-CH` (Swiss German)
- * "translation": {
- * "greeting": "Grüezi!"
- * }
- * }
- * }
- * ```
- * @param {String} locale The locale to translate to.
- * @returns {Object} The translation or an empty object if no matching
- * translation was found.
- */
- getJsonTranslation(locale) {
- // If there is no JSON resource
- if (this.jsonParameters === undefined) {
- return {};
- }
- // If locale is not set use the fallback locale
- locale = locale || this.pagesConfig.localizationFallbackLocale;
- // Get matching translation by locale, language or fallback locale
- const language = locale.split('-')[0];
- const resource = this.jsonParameters[locale] || this.jsonParameters[language] || this.jsonParameters[this.pagesConfig.localizationFallbackLocale] || {};
- const translation = resource.translation || {};
- return translation;
- }
- /**
- * Returns a translation from the JSON resource for a given locale with
- * placeholders filled in by given parameters.
- * @param {String} locale The locale to translate to.
- * @param {Object} params The parameters to fill into any placeholders
- * within the translations.
- * @returns {Object} The translation or an empty object if no matching
- * translation was found.
- */
- getJsonPlaceholders(locale, params = {}) {
- // If localization is disabled or there is no JSON resource
- if (!this.pagesConfig.enableLocalization || !this.pagesConfig.localizationJsonPath) {
- return {};
- }
- // Get JSON placeholders
- let placeholders = this.getJsonTranslation(locale);
- // Fill in any placeholders in the translation; this allows a translation
- // to contain default placeholders like {{appName}} which are filled here
- placeholders = JSON.stringify(placeholders);
- placeholders = _mustache.default.render(placeholders, params);
- placeholders = JSON.parse(placeholders);
- return placeholders;
- }
- /**
- * Creates a response with file content.
- * @param {String} path The path of the file to return.
- * @param {Object} [params={}] The parameters to be included in the response
- * header. These will also be used to fill placeholders.
- * @param {Object} [placeholders={}] The placeholders to fill in the content.
- * These will not be included in the response header.
- * @returns {Object} The Promise Router response.
- */
- async pageResponse(path, params = {}, placeholders = {}) {
- // Get file content
- let data;
- try {
- data = await this.readFile(path);
- } catch (e) {
- return this.notFound();
- }
- // Get config placeholders; can be an object, a function or an async function
- let configPlaceholders = typeof this.pagesConfig.placeholders === 'function' ? this.pagesConfig.placeholders(params) : Object.prototype.toString.call(this.pagesConfig.placeholders) === '[object Object]' ? this.pagesConfig.placeholders : {};
- if (configPlaceholders instanceof Promise) {
- configPlaceholders = await configPlaceholders;
- }
- // Fill placeholders
- const allPlaceholders = Object.assign({}, configPlaceholders, placeholders);
- const paramsAndPlaceholders = Object.assign({}, params, allPlaceholders);
- data = _mustache.default.render(data, paramsAndPlaceholders);
- // Add placeholders in header to allow parsing for programmatic use
- // of response, instead of having to parse the HTML content.
- const headers = Object.entries(params).reduce((m, p) => {
- if (p[1] !== undefined) {
- m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];
- }
- return m;
- }, {});
- return {
- text: data,
- headers: headers
- };
- }
- /**
- * Creates a response with file content.
- * @param {String} path The path of the file to return.
- * @returns {Object} The PromiseRouter response.
- */
- async fileResponse(path) {
- // Get file content
- let data;
- try {
- data = await this.readFile(path);
- } catch (e) {
- return this.notFound();
- }
- return {
- text: data
- };
- }
- /**
- * Reads and returns the content of a file at a given path. File reading to
- * serve content on the static route is only allowed from the pages
- * directory on downwards.
- * -----------------------------------------------------------------------
- * **WARNING:** All file reads in the PagesRouter must be executed by this
- * wrapper because it also detects and prevents common exploits.
- * -----------------------------------------------------------------------
- * @param {String} filePath The path to the file to read.
- * @returns {Promise<String>} The file content.
- */
- async readFile(filePath) {
- // Normalize path to prevent it from containing any directory changing
- // UNIX patterns which could expose the whole file system, e.g.
- // `http://example.com/parse/apps/../file.txt` requests a file outside
- // of the pages directory scope.
- const normalizedPath = _path.default.normalize(filePath);
- // Abort if the path is outside of the path directory scope
- if (!normalizedPath.startsWith(this.pagesPath)) {
- throw errors.fileOutsideAllowedScope;
- }
- return await _fs.promises.readFile(normalizedPath, 'utf-8');
- }
- /**
- * Loads a language resource JSON file that is used for translations.
- */
- loadJsonResource() {
- if (this.pagesConfig.localizationJsonPath === undefined) {
- return;
- }
- try {
- const json = require(_path.default.resolve('./', this.pagesConfig.localizationJsonPath));
- this.jsonParameters = json;
- } catch (e) {
- throw errors.jsonFailedFileLoading;
- }
- }
- /**
- * Extracts and returns the page default parameters from the Parse Server
- * configuration. These parameters are made accessible in every page served
- * by this router.
- * @param {Object} config The Parse Server configuration.
- * @returns {Object} The default parameters.
- */
- getDefaultParams(config) {
- return config ? {
- [pageParams.appId]: config.appId,
- [pageParams.appName]: config.appName,
- [pageParams.publicServerUrl]: config.publicServerURL
- } : {};
- }
- /**
- * Extracts and returns the locale from an express request.
- * @param {Object} req The express request.
- * @returns {String|undefined} The locale, or undefined if no locale was set.
- */
- getLocale(req) {
- const locale = (req.query || {})[pageParams.locale] || (req.body || {})[pageParams.locale] || (req.params || {})[pageParams.locale] || (req.headers || {})[pageParamHeaderPrefix + pageParams.locale];
- return locale;
- }
- /**
- * Creates a response with http redirect.
- * @param {Object} req The express request.
- * @param {String} path The path of the file to return.
- * @param {Object} params The query parameters to include.
- * @returns {Object} The Promise Router response.
- */
- async redirectResponse(url, params) {
- // Remove any parameters with undefined value
- params = Object.entries(params).reduce((m, p) => {
- if (p[1] !== undefined) {
- m[p[0]] = p[1];
- }
- return m;
- }, {});
- // Compose URL with parameters in query
- const location = new URL(url);
- Object.entries(params).forEach(p => location.searchParams.set(p[0], p[1]));
- const locationString = location.toString();
- // Add parameters to header to allow parsing for programmatic use
- // of response, instead of having to parse the HTML content.
- const headers = Object.entries(params).reduce((m, p) => {
- if (p[1] !== undefined) {
- m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];
- }
- return m;
- }, {});
- return {
- status: 303,
- location: locationString,
- headers: headers
- };
- }
- defaultPagePath(file) {
- return _path.default.join(this.pagesPath, file);
- }
- composePageUrl(file, publicServerUrl, locale) {
- let url = publicServerUrl;
- url += url.endsWith('/') ? '' : '/';
- url += this.pagesEndpoint + '/';
- url += locale === undefined ? '' : locale + '/';
- url += file;
- return url;
- }
- notFound() {
- return {
- text: 'Not found.',
- status: 404
- };
- }
- invalidRequest() {
- const error = new Error();
- error.status = 403;
- error.message = 'unauthorized';
- throw error;
- }
- /**
- * Sets the Parse Server configuration in the request object to make it
- * easily accessible throughtout request processing.
- * @param {Object} req The request.
- * @param {Boolean} failGracefully Is true if failing to set the config should
- * not result in an invalid request response. Default is `false`.
- */
- setConfig(req, failGracefully = false) {
- req.config = _Config.default.get(req.params.appId || req.query.appId);
- if (!req.config && !failGracefully) {
- this.invalidRequest();
- }
- return Promise.resolve();
- }
- mountPagesRoutes() {
- this.route('GET', `/${this.pagesEndpoint}/:appId/verify_email`, req => {
- this.setConfig(req);
- }, req => {
- return this.verifyEmail(req);
- });
- this.route('POST', `/${this.pagesEndpoint}/:appId/resend_verification_email`, req => {
- this.setConfig(req);
- }, req => {
- return this.resendVerificationEmail(req);
- });
- this.route('GET', `/${this.pagesEndpoint}/choose_password`, req => {
- this.setConfig(req);
- }, req => {
- return this.passwordReset(req);
- });
- this.route('POST', `/${this.pagesEndpoint}/:appId/request_password_reset`, req => {
- this.setConfig(req);
- }, req => {
- return this.resetPassword(req);
- });
- this.route('GET', `/${this.pagesEndpoint}/:appId/request_password_reset`, req => {
- this.setConfig(req);
- }, req => {
- return this.requestResetPassword(req);
- });
- }
- mountCustomRoutes() {
- for (const route of this.pagesConfig.customRoutes || []) {
- this.route(route.method, `/${this.pagesEndpoint}/:appId/${route.path}`, req => {
- this.setConfig(req);
- }, async req => {
- const {
- file,
- query = {}
- } = (await route.handler(req)) || {};
- // If route handler did not return a page send 404 response
- if (!file) {
- return this.notFound();
- }
- // Send page response
- const page = new _Page.default({
- id: file,
- defaultFile: file
- });
- return this.goToPage(req, page, query, false);
- });
- }
- }
- mountStaticRoute() {
- this.route('GET', `/${this.pagesEndpoint}/(*)?`, req => {
- this.setConfig(req, true);
- }, req => {
- return this.staticRoute(req);
- });
- }
- expressRouter() {
- const router = _express.default.Router();
- router.use('/', super.expressRouter());
- return router;
- }
- }
- exports.PagesRouter = PagesRouter;
- var _default = exports.default = PagesRouter;
- module.exports = {
- PagesRouter,
- pageParamHeaderPrefix,
- pageParams,
- pages
- };
- //# 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","params","resendVerificationEmail","body","publicServerURL","requestResetPassword","checkResetTokenValidity","applicationId","resetPassword","new_password","xhr","Parse","Error","USERNAME_MISSING","OTHER_CAUSE","PASSWORD_MISSING","updatePassword","Promise","success","err","result","status","response","page","responseType","redirect","forceRedirect","undefined","method","defaultParams","getDefaultParams","values","includes","notFound","assign","getLocale","defaultPath","defaultPagePath","defaultUrl","composePageUrl","customUrl","customUrls","Utils","isPath","redirectResponse","placeholders","enableLocalization","localizationJsonPath","getJsonPlaceholders","getLocalizedPath","subdir","pageResponse","staticRoute","relativePath","absolutePath","endsWith","fileResponse","getJsonTranslation","jsonParameters","localizationFallbackLocale","language","split","resource","translation","JSON","stringify","mustache","render","parse","data","readFile","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 { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!token || !username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n    return userController.verifyEmail(username, token).then(\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationSuccess, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationLinkExpired, params);\n      }\n    );\n  }\n\n  resendVerificationEmail(req) {\n    const config = req.config;\n    const username = req.body.username;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n\n    return userController.resendVerificationEmail(username, req).then(\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendSuccess);\n      },\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendFail);\n      }\n    );\n  }\n\n  passwordReset(req) {\n    const config = req.config;\n    const params = {\n      [pageParams.appId]: req.params.appId,\n      [pageParams.appName]: config.appName,\n      [pageParams.token]: req.query.token,\n      [pageParams.username]: req.query.username,\n      [pageParams.publicServerUrl]: config.publicServerURL,\n    };\n    return this.goToPage(req, pages.passwordReset, params);\n  }\n\n  requestResetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!username || !token) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    return config.userController.checkResetTokenValidity(username, token).then(\n      () => {\n        const params = {\n          [pageParams.token]: token,\n          [pageParams.username]: username,\n          [pageParams.appId]: config.applicationId,\n          [pageParams.appName]: config.appName,\n        };\n        return this.goToPage(req, pages.passwordReset, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.passwordResetLinkInvalid, params);\n      }\n    );\n  }\n\n  resetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, new_password, token: rawToken } = req.body;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if ((!username || !token || !new_password) && req.xhr === false) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    if (!username) {\n      throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'Missing username');\n    }\n\n    if (!token) {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'Missing token');\n    }\n\n    if (!new_password) {\n      throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'Missing password');\n    }\n\n    return config.userController\n      .updatePassword(username, token, new_password)\n      .then(\n        () => {\n          return Promise.resolve({\n            success: true,\n          });\n        },\n        err => {\n          return Promise.resolve({\n            success: false,\n            err,\n          });\n        }\n      )\n      .then(result => {\n        if (req.xhr) {\n          if (result.success) {\n            return Promise.resolve({\n              status: 200,\n              response: 'Password successfully reset',\n            });\n          }\n          if (result.err) {\n            throw new Parse.Error(Parse.Error.OTHER_CAUSE, `${result.err}`);\n          }\n        }\n\n        const query = result.success\n          ? {\n            [pageParams.username]: username,\n          }\n          : {\n            [pageParams.username]: username,\n            [pageParams.token]: token,\n            [pageParams.appId]: config.applicationId,\n            [pageParams.error]: result.err,\n            [pageParams.appName]: config.appName,\n          };\n        const page = result.success ? pages.passwordResetSuccess : pages.passwordReset;\n\n        return this.goToPage(req, page, query, false);\n      });\n  }\n\n  /**\n   * Returns page content if the page is a local file or returns a\n   * redirect to a custom page.\n   * @param {Object} req The express request.\n   * @param {Page} page The page to go to.\n   * @param {Object} [params={}] The query parameters to attach to the URL in case of\n   * HTTP redirect responses for POST requests, or the placeholders to fill into\n   * the response content in case of HTTP content responses for GET requests.\n   * @param {Boolean} [responseType] Is true if a redirect response should be forced,\n   * false if a content response should be forced, undefined if the response type\n   * should depend on the request type by default:\n   * - GET request -> content response\n   * - POST request -> redirect response (PRG pattern)\n   * @returns {Promise<Object>} The PromiseRouter response.\n   */\n  goToPage(req, page, params = {}, responseType) {\n    const config = req.config;\n\n    // Determine redirect either by force, response setting or request method\n    const redirect = config.pages.forceRedirect\n      ? true\n      : responseType !== undefined\n        ? responseType\n        : req.method == 'POST';\n\n    // Include default parameters\n    const defaultParams = this.getDefaultParams(config);\n    if (Object.values(defaultParams).includes(undefined)) {\n      return this.notFound();\n    }\n    params = Object.assign(params, defaultParams);\n\n    // Add locale to params to ensure it is passed on with every request;\n    // that means, once a locale is set, it is passed on to any follow-up page,\n    // e.g. request_password_reset -> password_reset -> password_reset_success\n    const locale = this.getLocale(req);\n    params[pageParams.locale] = locale;\n\n    // Compose paths and URLs\n    const defaultFile = page.defaultFile;\n    const defaultPath = this.defaultPagePath(defaultFile);\n    const defaultUrl = this.composePageUrl(defaultFile, config.publicServerURL);\n\n    // If custom URL is set redirect to it without localization\n    const customUrl = config.pages.customUrls[page.id];\n    if (customUrl && !Utils.isPath(customUrl)) {\n      return this.redirectResponse(customUrl, params);\n    }\n\n    // Get JSON placeholders\n    let placeholders = {};\n    if (config.pages.enableLocalization && config.pages.localizationJsonPath) {\n      placeholders = this.getJsonPlaceholders(locale, params);\n    }\n\n    // Send response\n    if (config.pages.enableLocalization && locale) {\n      return Utils.getLocalizedPath(defaultPath, locale).then(({ path, subdir }) =>\n        redirect\n          ? this.redirectResponse(\n            this.composePageUrl(defaultFile, config.publicServerURL, subdir),\n            params\n          )\n          : this.pageResponse(path, params, placeholders)\n      );\n    } else {\n      return redirect\n        ? this.redirectResponse(defaultUrl, params)\n        : this.pageResponse(defaultPath, params, placeholders);\n    }\n  }\n\n  /**\n   * Serves a request to a static resource and localizes the resource if it\n   * is a HTML file.\n   * @param {Object} req The request object.\n   * @returns {Promise<Object>} The response.\n   */\n  staticRoute(req) {\n    // Get requested path\n    const relativePath = req.params[0];\n\n    // Resolve requested path to absolute path\n    const absolutePath = path.resolve(this.pagesPath, relativePath);\n\n    // If the requested file is not a HTML file send its raw content\n    if (!absolutePath || !absolutePath.endsWith('.html')) {\n      return this.fileResponse(absolutePath);\n    }\n\n    // Get parameters\n    const params = this.getDefaultParams(req.config);\n    const locale = this.getLocale(req);\n    if (locale) {\n      params.locale = locale;\n    }\n\n    // Get JSON placeholders\n    const placeholders = this.getJsonPlaceholders(locale, params);\n\n    return this.pageResponse(absolutePath, params, placeholders);\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale. The JSON\n   * resource is parsed according to i18next syntax.\n   *\n   * Example JSON content:\n   * ```js\n   *  {\n   *    \"en\": {               // resource for language `en` (English)\n   *      \"translation\": {\n   *        \"greeting\": \"Hello!\"\n   *      }\n   *    },\n   *    \"de\": {               // resource for language `de` (German)\n   *      \"translation\": {\n   *        \"greeting\": \"Hallo!\"\n   *      }\n   *    }\n   *    \"de-CH\": {            // resource for locale `de-CH` (Swiss German)\n   *      \"translation\": {\n   *        \"greeting\": \"Grüezi!\"\n   *      }\n   *    }\n   *  }\n   * ```\n   * @param {String} locale The locale to translate to.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonTranslation(locale) {\n    // If there is no JSON resource\n    if (this.jsonParameters === undefined) {\n      return {};\n    }\n\n    // If locale is not set use the fallback locale\n    locale = locale || this.pagesConfig.localizationFallbackLocale;\n\n    // Get matching translation by locale, language or fallback locale\n    const language = locale.split('-')[0];\n    const resource =\n      this.jsonParameters[locale] ||\n      this.jsonParameters[language] ||\n      this.jsonParameters[this.pagesConfig.localizationFallbackLocale] ||\n      {};\n    const translation = resource.translation || {};\n    return translation;\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale with\n   * placeholders filled in by given parameters.\n   * @param {String} locale The locale to translate to.\n   * @param {Object} params The parameters to fill into any placeholders\n   * within the translations.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonPlaceholders(locale, params = {}) {\n    // If localization is disabled or there is no JSON resource\n    if (!this.pagesConfig.enableLocalization || !this.pagesConfig.localizationJsonPath) {\n      return {};\n    }\n\n    // Get JSON placeholders\n    let placeholders = this.getJsonTranslation(locale);\n\n    // Fill in any placeholders in the translation; this allows a translation\n    // to contain default placeholders like {{appName}} which are filled here\n    placeholders = JSON.stringify(placeholders);\n    placeholders = mustache.render(placeholders, params);\n    placeholders = JSON.parse(placeholders);\n\n    return placeholders;\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @param {Object} [params={}] The parameters to be included in the response\n   * header. These will also be used to fill placeholders.\n   * @param {Object} [placeholders={}] The placeholders to fill in the content.\n   * These will not be included in the response header.\n   * @returns {Object} The Promise Router response.\n   */\n  async pageResponse(path, params = {}, placeholders = {}) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    // Get config placeholders; can be an object, a function or an async function\n    let configPlaceholders =\n      typeof this.pagesConfig.placeholders === 'function'\n        ? this.pagesConfig.placeholders(params)\n        : Object.prototype.toString.call(this.pagesConfig.placeholders) === '[object Object]'\n          ? this.pagesConfig.placeholders\n          : {};\n    if (configPlaceholders instanceof Promise) {\n      configPlaceholders = await configPlaceholders;\n    }\n\n    // Fill placeholders\n    const allPlaceholders = Object.assign({}, configPlaceholders, placeholders);\n    const paramsAndPlaceholders = Object.assign({}, params, allPlaceholders);\n    data = mustache.render(data, paramsAndPlaceholders);\n\n    // Add placeholders in header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return { text: data, headers: headers };\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @returns {Object} The PromiseRouter response.\n   */\n  async fileResponse(path) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    return { text: data };\n  }\n\n  /**\n   * Reads and returns the content of a file at a given path. File reading to\n   * serve content on the static route is only allowed from the pages\n   * directory on downwards.\n   * -----------------------------------------------------------------------\n   * **WARNING:** All file reads in the PagesRouter must be executed by this\n   * wrapper because it also detects and prevents common exploits.\n   * -----------------------------------------------------------------------\n   * @param {String} filePath The path to the file to read.\n   * @returns {Promise<String>} The file content.\n   */\n  async readFile(filePath) {\n    // Normalize path to prevent it from containing any directory changing\n    // UNIX patterns which could expose the whole file system, e.g.\n    // `http://example.com/parse/apps/../file.txt` requests a file outside\n    // of the pages directory scope.\n    const normalizedPath = path.normalize(filePath);\n\n    // Abort if the path is outside of the path directory scope\n    if (!normalizedPath.startsWith(this.pagesPath)) {\n      throw errors.fileOutsideAllowedScope;\n    }\n\n    return await fs.readFile(normalizedPath, 'utf-8');\n  }\n\n  /**\n   * Loads a language resource JSON file that is used for translations.\n   */\n  loadJsonResource() {\n    if (this.pagesConfig.localizationJsonPath === undefined) {\n      return;\n    }\n    try {\n      const json = require(path.resolve('./', this.pagesConfig.localizationJsonPath));\n      this.jsonParameters = json;\n    } catch (e) {\n      throw errors.jsonFailedFileLoading;\n    }\n  }\n\n  /**\n   * Extracts and returns the page default parameters from the Parse Server\n   * configuration. These parameters are made accessible in every page served\n   * by this router.\n   * @param {Object} config The Parse Server configuration.\n   * @returns {Object} The default parameters.\n   */\n  getDefaultParams(config) {\n    return config\n      ? {\n        [pageParams.appId]: config.appId,\n        [pageParams.appName]: config.appName,\n        [pageParams.publicServerUrl]: config.publicServerURL,\n      }\n      : {};\n  }\n\n  /**\n   * Extracts and returns the locale from an express request.\n   * @param {Object} req The express request.\n   * @returns {String|undefined} The locale, or undefined if no locale was set.\n   */\n  getLocale(req) {\n    const locale =\n      (req.query || {})[pageParams.locale] ||\n      (req.body || {})[pageParams.locale] ||\n      (req.params || {})[pageParams.locale] ||\n      (req.headers || {})[pageParamHeaderPrefix + pageParams.locale];\n    return locale;\n  }\n\n  /**\n   * Creates a response with http redirect.\n   * @param {Object} req The express request.\n   * @param {String} path The path of the file to return.\n   * @param {Object} params The query parameters to include.\n   * @returns {Object} The Promise Router response.\n   */\n  async redirectResponse(url, params) {\n    // Remove any parameters with undefined value\n    params = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[p[0]] = p[1];\n      }\n      return m;\n    }, {});\n\n    // Compose URL with parameters in query\n    const location = new URL(url);\n    Object.entries(params).forEach(p => location.searchParams.set(p[0], p[1]));\n    const locationString = location.toString();\n\n    // Add parameters to header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return {\n      status: 303,\n      location: locationString,\n      headers: headers,\n    };\n  }\n\n  defaultPagePath(file) {\n    return path.join(this.pagesPath, file);\n  }\n\n  composePageUrl(file, publicServerUrl, locale) {\n    let url = publicServerUrl;\n    url += url.endsWith('/') ? '' : '/';\n    url += this.pagesEndpoint + '/';\n    url += locale === undefined ? '' : locale + '/';\n    url += file;\n    return url;\n  }\n\n  notFound() {\n    return {\n      text: 'Not found.',\n      status: 404,\n    };\n  }\n\n  invalidRequest() {\n    const error = new Error();\n    error.status = 403;\n    error.message = 'unauthorized';\n    throw error;\n  }\n\n  /**\n   * Sets the Parse Server configuration in the request object to make it\n   * easily accessible throughtout request processing.\n   * @param {Object} req The request.\n   * @param {Boolean} failGracefully Is true if failing to set the config should\n   * not result in an invalid request response. Default is `false`.\n   */\n  setConfig(req, failGracefully = false) {\n    req.config = Config.get(req.params.appId || req.query.appId);\n    if (!req.config && !failGracefully) {\n      this.invalidRequest();\n    }\n    return Promise.resolve();\n  }\n\n  mountPagesRoutes() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/verify_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.verifyEmail(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/resend_verification_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resendVerificationEmail(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/choose_password`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.passwordReset(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resetPassword(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.requestResetPassword(req);\n      }\n    );\n  }\n\n  mountCustomRoutes() {\n    for (const route of this.pagesConfig.customRoutes || []) {\n      this.route(\n        route.method,\n        `/${this.pagesEndpoint}/:appId/${route.path}`,\n        req => {\n          this.setConfig(req);\n        },\n        async req => {\n          const { file, query = {} } = (await route.handler(req)) || {};\n\n          // If route handler did not return a page send 404 response\n          if (!file) {\n            return this.notFound();\n          }\n\n          // Send page response\n          const page = new Page({ id: file, defaultFile: file });\n          return this.goToPage(req, page, query, false);\n        }\n      );\n    }\n  }\n\n  mountStaticRoute() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/(*)?`,\n      req => {\n        this.setConfig(req, true);\n      },\n      req => {\n        return this.staticRoute(req);\n      }\n    );\n  }\n\n  expressRouter() {\n    const router = express.Router();\n    router.use('/', super.expressRouter());\n    return router;\n  }\n}\n\nexport default PagesRouter;\nmodule.exports = {\n  PagesRouter,\n  pageParamHeaderPrefix,\n  pageParams,\n  pages,\n};\n"],"mappings":"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","ignoreList":[]}
|