"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.addRateLimit = exports.DEFAULT_ALLOWED_HEADERS = void 0; exports.allowCrossDomain = allowCrossDomain; exports.allowMethodOverride = allowMethodOverride; exports.checkIp = void 0; exports.enforceMasterKeyAccess = enforceMasterKeyAccess; exports.handleParseErrors = handleParseErrors; exports.handleParseHeaders = handleParseHeaders; exports.handleParseSession = void 0; exports.promiseEnforceMasterKeyAccess = promiseEnforceMasterKeyAccess; exports.promiseEnsureIdempotency = promiseEnsureIdempotency; var _cache = _interopRequireDefault(require("./cache")); var _node = _interopRequireDefault(require("parse/node")); var _Auth = _interopRequireDefault(require("./Auth")); var _Config = _interopRequireDefault(require("./Config")); var _ClientSDK = _interopRequireDefault(require("./ClientSDK")); var _logger = _interopRequireDefault(require("./logger")); var _rest = _interopRequireDefault(require("./rest")); var _MongoStorageAdapter = _interopRequireDefault(require("./Adapters/Storage/Mongo/MongoStorageAdapter")); var _PostgresStorageAdapter = _interopRequireDefault(require("./Adapters/Storage/Postgres/PostgresStorageAdapter")); var _expressRateLimit = _interopRequireDefault(require("express-rate-limit")); var _Definitions = require("./Options/Definitions"); var _pathToRegexp = require("path-to-regexp"); var _rateLimitRedis = _interopRequireDefault(require("rate-limit-redis")); var _redis = require("redis"); var _net = require("net"); function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; } const DEFAULT_ALLOWED_HEADERS = exports.DEFAULT_ALLOWED_HEADERS = 'X-Parse-Master-Key, X-Parse-REST-API-Key, X-Parse-Javascript-Key, X-Parse-Application-Id, X-Parse-Client-Version, X-Parse-Session-Token, X-Requested-With, X-Parse-Revocable-Session, X-Parse-Request-Id, Content-Type, Pragma, Cache-Control'; const getMountForRequest = function (req) { const mountPathLength = req.originalUrl.length - req.url.length; const mountPath = req.originalUrl.slice(0, mountPathLength); return req.protocol + '://' + req.get('host') + mountPath; }; const getBlockList = (ipRangeList, store) => { if (store.get('blockList')) return store.get('blockList'); const blockList = new _net.BlockList(); ipRangeList.forEach(fullIp => { if (fullIp === '::/0' || fullIp === '::') { store.set('allowAllIpv6', true); return; } if (fullIp === '0.0.0.0/0' || fullIp === '0.0.0.0') { store.set('allowAllIpv4', true); return; } const [ip, mask] = fullIp.split('/'); if (!mask) { blockList.addAddress(ip, (0, _net.isIPv4)(ip) ? 'ipv4' : 'ipv6'); } else { blockList.addSubnet(ip, Number(mask), (0, _net.isIPv4)(ip) ? 'ipv4' : 'ipv6'); } }); store.set('blockList', blockList); return blockList; }; const checkIp = (ip, ipRangeList, store) => { const incomingIpIsV4 = (0, _net.isIPv4)(ip); const blockList = getBlockList(ipRangeList, store); if (store.get(ip)) return true; if (store.get('allowAllIpv4') && incomingIpIsV4) return true; if (store.get('allowAllIpv6') && !incomingIpIsV4) return true; const result = blockList.check(ip, incomingIpIsV4 ? 'ipv4' : 'ipv6'); // If the ip is in the list, we store the result in the store // so we have a optimized path for the next request if (ipRangeList.includes(ip) && result) { store.set(ip, result); } return result; }; // Checks that the request is authorized for this app and checks user // auth too. // The bodyparser should run before this middleware. // Adds info to the request: // req.config - the Config for this app // req.auth - the Auth for this request exports.checkIp = checkIp; function handleParseHeaders(req, res, next) { var mount = getMountForRequest(req); let context = {}; if (req.get('X-Parse-Cloud-Context') != null) { try { context = JSON.parse(req.get('X-Parse-Cloud-Context')); if (Object.prototype.toString.call(context) !== '[object Object]') { throw 'Context is not an object'; } } catch (e) { return malformedContext(req, res); } } var info = { appId: req.get('X-Parse-Application-Id'), sessionToken: req.get('X-Parse-Session-Token'), masterKey: req.get('X-Parse-Master-Key'), maintenanceKey: req.get('X-Parse-Maintenance-Key'), installationId: req.get('X-Parse-Installation-Id'), clientKey: req.get('X-Parse-Client-Key'), javascriptKey: req.get('X-Parse-Javascript-Key'), dotNetKey: req.get('X-Parse-Windows-Key'), restAPIKey: req.get('X-Parse-REST-API-Key'), clientVersion: req.get('X-Parse-Client-Version'), context: context }; var basicAuth = httpAuth(req); if (basicAuth) { var basicAuthAppId = basicAuth.appId; if (_cache.default.get(basicAuthAppId)) { info.appId = basicAuthAppId; info.masterKey = basicAuth.masterKey || info.masterKey; info.javascriptKey = basicAuth.javascriptKey || info.javascriptKey; } } if (req.body) { // Unity SDK sends a _noBody key which needs to be removed. // Unclear at this point if action needs to be taken. delete req.body._noBody; } var fileViaJSON = false; if (!info.appId || !_cache.default.get(info.appId)) { // See if we can find the app id on the body. if (req.body instanceof Buffer) { // The only chance to find the app id is if this is a file // upload that actually is a JSON body. So try to parse it. // https://github.com/parse-community/parse-server/issues/6589 // It is also possible that the client is trying to upload a file but forgot // to provide x-parse-app-id in header and parse a binary file will fail try { req.body = JSON.parse(req.body); } catch (e) { return invalidRequest(req, res); } fileViaJSON = true; } if (req.body) { delete req.body._RevocableSession; } if (req.body && req.body._ApplicationId && _cache.default.get(req.body._ApplicationId) && (!info.masterKey || _cache.default.get(req.body._ApplicationId).masterKey === info.masterKey)) { info.appId = req.body._ApplicationId; info.javascriptKey = req.body._JavaScriptKey || ''; delete req.body._ApplicationId; delete req.body._JavaScriptKey; // TODO: test that the REST API formats generated by the other // SDKs are handled ok if (req.body._ClientVersion) { info.clientVersion = req.body._ClientVersion; delete req.body._ClientVersion; } if (req.body._InstallationId) { info.installationId = req.body._InstallationId; delete req.body._InstallationId; } if (req.body._SessionToken) { info.sessionToken = req.body._SessionToken; delete req.body._SessionToken; } if (req.body._MasterKey) { info.masterKey = req.body._MasterKey; delete req.body._MasterKey; } if (req.body._context) { if (req.body._context instanceof Object) { info.context = req.body._context; } else { try { info.context = JSON.parse(req.body._context); if (Object.prototype.toString.call(info.context) !== '[object Object]') { throw 'Context is not an object'; } } catch (e) { return malformedContext(req, res); } } delete req.body._context; } if (req.body._ContentType) { req.headers['content-type'] = req.body._ContentType; delete req.body._ContentType; } } else { return invalidRequest(req, res); } } if (info.sessionToken && typeof info.sessionToken !== 'string') { info.sessionToken = info.sessionToken.toString(); } if (info.clientVersion) { info.clientSDK = _ClientSDK.default.fromString(info.clientVersion); } if (fileViaJSON) { req.fileData = req.body.fileData; // We need to repopulate req.body with a buffer var base64 = req.body.base64; req.body = Buffer.from(base64, 'base64'); } const clientIp = getClientIp(req); const config = _Config.default.get(info.appId, mount); if (config.state && config.state !== 'ok') { res.status(500); res.json({ code: _node.default.Error.INTERNAL_SERVER_ERROR, error: `Invalid server state: ${config.state}` }); return; } info.app = _cache.default.get(info.appId); req.config = config; req.config.headers = req.headers || {}; req.config.ip = clientIp; req.info = info; const isMaintenance = req.config.maintenanceKey && info.maintenanceKey === req.config.maintenanceKey; if (isMaintenance) { var _req$config; if (checkIp(clientIp, req.config.maintenanceKeyIps || [], req.config.maintenanceKeyIpsStore)) { req.auth = new _Auth.default.Auth({ config: req.config, installationId: info.installationId, isMaintenance: true }); next(); return; } const log = ((_req$config = req.config) === null || _req$config === void 0 ? void 0 : _req$config.loggerController) || _logger.default; log.error(`Request using maintenance key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'maintenanceKeyIps'.`); } let isMaster = info.masterKey === req.config.masterKey; if (isMaster && !checkIp(clientIp, req.config.masterKeyIps || [], req.config.masterKeyIpsStore)) { var _req$config2; const log = ((_req$config2 = req.config) === null || _req$config2 === void 0 ? void 0 : _req$config2.loggerController) || _logger.default; log.error(`Request using master key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'masterKeyIps'.`); isMaster = false; const error = new Error(); error.status = 403; error.message = `unauthorized`; throw error; } if (isMaster) { req.auth = new _Auth.default.Auth({ config: req.config, installationId: info.installationId, isMaster: true }); return handleRateLimit(req, res, next); } var isReadOnlyMaster = info.masterKey === req.config.readOnlyMasterKey; if (typeof req.config.readOnlyMasterKey != 'undefined' && req.config.readOnlyMasterKey && isReadOnlyMaster) { req.auth = new _Auth.default.Auth({ config: req.config, installationId: info.installationId, isMaster: true, isReadOnly: true }); return handleRateLimit(req, res, next); } // Client keys are not required in parse-server, but if any have been configured in the server, validate them // to preserve original behavior. const keys = ['clientKey', 'javascriptKey', 'dotNetKey', 'restAPIKey']; const oneKeyConfigured = keys.some(function (key) { return req.config[key] !== undefined; }); const oneKeyMatches = keys.some(function (key) { return req.config[key] !== undefined && info[key] === req.config[key]; }); if (oneKeyConfigured && !oneKeyMatches) { return invalidRequest(req, res); } if (req.url == '/login') { delete info.sessionToken; } if (req.userFromJWT) { req.auth = new _Auth.default.Auth({ config: req.config, installationId: info.installationId, isMaster: false, user: req.userFromJWT }); return handleRateLimit(req, res, next); } if (!info.sessionToken) { req.auth = new _Auth.default.Auth({ config: req.config, installationId: info.installationId, isMaster: false }); } handleRateLimit(req, res, next); } const handleRateLimit = async (req, res, next) => { const rateLimits = req.config.rateLimits || []; try { await Promise.all(rateLimits.map(async limit => { const pathExp = new RegExp(limit.path); if (pathExp.test(req.url)) { await limit.handler(req, res, err => { if (err) { if (err.code === _node.default.Error.CONNECTION_FAILED) { throw err; } req.config.loggerController.error('An unknown error occured when attempting to apply the rate limiter: ', err); } }); } })); } catch (error) { res.status(429); res.json({ code: _node.default.Error.CONNECTION_FAILED, error: error.message }); return; } next(); }; const handleParseSession = async (req, res, next) => { try { const info = req.info; if (req.auth || req.url === '/sessions/me') { next(); return; } let requestAuth = null; if (info.sessionToken && req.url === '/upgradeToRevocableSession' && info.sessionToken.indexOf('r:') != 0) { requestAuth = await _Auth.default.getAuthForLegacySessionToken({ config: req.config, installationId: info.installationId, sessionToken: info.sessionToken }); } else { requestAuth = await _Auth.default.getAuthForSessionToken({ config: req.config, installationId: info.installationId, sessionToken: info.sessionToken }); } req.auth = requestAuth; next(); } catch (error) { if (error instanceof _node.default.Error) { next(error); return; } // TODO: Determine the correct error scenario. req.config.loggerController.error('error getting auth for sessionToken', error); throw new _node.default.Error(_node.default.Error.UNKNOWN_ERROR, error); } }; exports.handleParseSession = handleParseSession; function getClientIp(req) { return req.ip; } function httpAuth(req) { if (!(req.req || req).headers.authorization) return; var header = (req.req || req).headers.authorization; var appId, masterKey, javascriptKey; // parse header var authPrefix = 'basic '; var match = header.toLowerCase().indexOf(authPrefix); if (match == 0) { var encodedAuth = header.substring(authPrefix.length, header.length); var credentials = decodeBase64(encodedAuth).split(':'); if (credentials.length == 2) { appId = credentials[0]; var key = credentials[1]; var jsKeyPrefix = 'javascript-key='; var matchKey = key.indexOf(jsKeyPrefix); if (matchKey == 0) { javascriptKey = key.substring(jsKeyPrefix.length, key.length); } else { masterKey = key; } } } return { appId: appId, masterKey: masterKey, javascriptKey: javascriptKey }; } function decodeBase64(str) { return Buffer.from(str, 'base64').toString(); } function allowCrossDomain(appId) { return (req, res, next) => { const config = _Config.default.get(appId, getMountForRequest(req)); let allowHeaders = DEFAULT_ALLOWED_HEADERS; if (config && config.allowHeaders) { allowHeaders += `, ${config.allowHeaders.join(', ')}`; } const baseOrigins = typeof (config === null || config === void 0 ? void 0 : config.allowOrigin) === 'string' ? [config.allowOrigin] : (config === null || config === void 0 ? void 0 : config.allowOrigin) ?? ['*']; const requestOrigin = req.headers.origin; const allowOrigins = requestOrigin && baseOrigins.includes(requestOrigin) ? requestOrigin : baseOrigins[0]; res.header('Access-Control-Allow-Origin', allowOrigins); res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS'); res.header('Access-Control-Allow-Headers', allowHeaders); res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id'); // intercept OPTIONS method if ('OPTIONS' == req.method) { res.sendStatus(200); } else { next(); } }; } function allowMethodOverride(req, res, next) { if (req.method === 'POST' && req.body._method) { req.originalMethod = req.method; req.method = req.body._method; delete req.body._method; } next(); } function handleParseErrors(err, req, res, next) { const log = req.config && req.config.loggerController || _logger.default; if (err instanceof _node.default.Error) { if (req.config && req.config.enableExpressErrorHandler) { return next(err); } let httpStatus; // TODO: fill out this mapping switch (err.code) { case _node.default.Error.INTERNAL_SERVER_ERROR: httpStatus = 500; break; case _node.default.Error.OBJECT_NOT_FOUND: httpStatus = 404; break; default: httpStatus = 400; } res.status(httpStatus); res.json({ code: err.code, error: err.message }); log.error('Parse error: ', err); } else if (err.status && err.message) { res.status(err.status); res.json({ error: err.message }); if (!(process && process.env.TESTING)) { next(err); } } else { log.error('Uncaught internal server error.', err, err.stack); res.status(500); res.json({ code: _node.default.Error.INTERNAL_SERVER_ERROR, message: 'Internal server error.' }); if (!(process && process.env.TESTING)) { next(err); } } } function enforceMasterKeyAccess(req, res, next) { if (!req.auth.isMaster) { res.status(403); res.end('{"error":"unauthorized: master key is required"}'); return; } next(); } function promiseEnforceMasterKeyAccess(request) { if (!request.auth.isMaster) { const error = new Error(); error.status = 403; error.message = 'unauthorized: master key is required'; throw error; } return Promise.resolve(); } const addRateLimit = (route, config, cloud) => { if (typeof config === 'string') { config = _Config.default.get(config); } for (const key in route) { if (!_Definitions.RateLimitOptions[key]) { throw `Invalid rate limit option "${key}"`; } } if (!config.rateLimits) { config.rateLimits = []; } const redisStore = { connectionPromise: Promise.resolve(), store: null }; if (route.redisUrl) { const client = (0, _redis.createClient)({ url: route.redisUrl }); redisStore.connectionPromise = async () => { if (client.isOpen) { return; } try { await client.connect(); } catch (e) { var _config; const log = ((_config = config) === null || _config === void 0 ? void 0 : _config.loggerController) || _logger.default; log.error(`Could not connect to redisURL in rate limit: ${e}`); } }; redisStore.connectionPromise(); redisStore.store = new _rateLimitRedis.default({ sendCommand: async (...args) => { await redisStore.connectionPromise(); return client.sendCommand(args); } }); } let transformPath = route.requestPath.split('/*').join('/(.*)'); if (transformPath === '*') { transformPath = '(.*)'; } config.rateLimits.push({ path: (0, _pathToRegexp.pathToRegexp)(transformPath), handler: (0, _expressRateLimit.default)({ windowMs: route.requestTimeWindow, max: route.requestCount, message: route.errorResponseMessage || _Definitions.RateLimitOptions.errorResponseMessage.default, handler: (request, response, next, options) => { throw { code: _node.default.Error.CONNECTION_FAILED, message: options.message }; }, skip: request => { var _request$auth; if (request.ip === '127.0.0.1' && !route.includeInternalRequests) { return true; } if (route.includeMasterKey) { return false; } if (route.requestMethods) { if (Array.isArray(route.requestMethods)) { if (!route.requestMethods.includes(request.method)) { return true; } } else { const regExp = new RegExp(route.requestMethods); if (!regExp.test(request.method)) { return true; } } } return (_request$auth = request.auth) === null || _request$auth === void 0 ? void 0 : _request$auth.isMaster; }, keyGenerator: async request => { if (route.zone === _node.default.Server.RateLimitZone.global) { return request.config.appId; } const token = request.info.sessionToken; if (route.zone === _node.default.Server.RateLimitZone.session && token) { return token; } if (route.zone === _node.default.Server.RateLimitZone.user && token) { var _request$auth2; if (!request.auth) { await new Promise(resolve => handleParseSession(request, null, resolve)); } if ((_request$auth2 = request.auth) !== null && _request$auth2 !== void 0 && (_request$auth2 = _request$auth2.user) !== null && _request$auth2 !== void 0 && _request$auth2.id && request.zone === 'user') { return request.auth.user.id; } } return request.config.ip; }, store: redisStore.store }), cloud }); _Config.default.put(config); }; /** * Deduplicates a request to ensure idempotency. Duplicates are determined by the request ID * in the request header. If a request has no request ID, it is executed anyway. * @param {*} req The request to evaluate. * @returns Promise<{}> */ exports.addRateLimit = addRateLimit; function promiseEnsureIdempotency(req) { // Enable feature only for MongoDB if (!(req.config.database.adapter instanceof _MongoStorageAdapter.default || req.config.database.adapter instanceof _PostgresStorageAdapter.default)) { return Promise.resolve(); } // Get parameters const config = req.config; const requestId = ((req || {}).headers || {})['x-parse-request-id']; const { paths, ttl } = config.idempotencyOptions; if (!requestId || !config.idempotencyOptions) { return Promise.resolve(); } // Request path may contain trailing slashes, depending on the original request, so remove // leading and trailing slashes to make it easier to specify paths in the configuration const reqPath = req.path.replace(/^\/|\/$/, ''); // Determine whether idempotency is enabled for current request path let match = false; for (const path of paths) { // Assume one wants a path to always match from the beginning to prevent any mistakes const regex = new RegExp(path.charAt(0) === '^' ? path : '^' + path); if (reqPath.match(regex)) { match = true; break; } } if (!match) { return Promise.resolve(); } // Try to store request const expiryDate = new Date(new Date().setSeconds(new Date().getSeconds() + ttl)); return _rest.default.create(config, _Auth.default.master(config), '_Idempotency', { reqId: requestId, expire: _node.default._encode(expiryDate) }).catch(e => { if (e.code == _node.default.Error.DUPLICATE_VALUE) { throw new _node.default.Error(_node.default.Error.DUPLICATE_REQUEST, 'Duplicate request'); } throw e; }); } function invalidRequest(req, res) { res.status(403); res.end('{"error":"unauthorized"}'); } function malformedContext(req, res) { res.status(400); res.json({ code: _node.default.Error.INVALID_JSON, error: 'Invalid object for context.' }); } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_cache","_interopRequireDefault","require","_node","_Auth","_Config","_ClientSDK","_logger","_rest","_MongoStorageAdapter","_PostgresStorageAdapter","_expressRateLimit","_Definitions","_pathToRegexp","_rateLimitRedis","_redis","_net","e","__esModule","default","DEFAULT_ALLOWED_HEADERS","exports","getMountForRequest","req","mountPathLength","originalUrl","length","url","mountPath","slice","protocol","get","getBlockList","ipRangeList","store","blockList","BlockList","forEach","fullIp","set","ip","mask","split","addAddress","isIPv4","addSubnet","Number","checkIp","incomingIpIsV4","result","check","includes","handleParseHeaders","res","next","mount","context","JSON","parse","Object","prototype","toString","call","malformedContext","info","appId","sessionToken","masterKey","maintenanceKey","installationId","clientKey","javascriptKey","dotNetKey","restAPIKey","clientVersion","basicAuth","httpAuth","basicAuthAppId","AppCache","body","_noBody","fileViaJSON","Buffer","invalidRequest","_RevocableSession","_ApplicationId","_JavaScriptKey","_ClientVersion","_InstallationId","_SessionToken","_MasterKey","_context","_ContentType","headers","clientSDK","ClientSDK","fromString","fileData","base64","from","clientIp","getClientIp","config","Config","state","status","json","code","Parse","Error","INTERNAL_SERVER_ERROR","error","app","isMaintenance","_req$config","maintenanceKeyIps","maintenanceKeyIpsStore","auth","Auth","log","loggerController","defaultLogger","isMaster","masterKeyIps","masterKeyIpsStore","_req$config2","message","handleRateLimit","isReadOnlyMaster","readOnlyMasterKey","isReadOnly","keys","oneKeyConfigured","some","key","undefined","oneKeyMatches","userFromJWT","user","rateLimits","Promise","all","map","limit","pathExp","RegExp","path","test","handler","err","CONNECTION_FAILED","handleParseSession","requestAuth","indexOf","getAuthForLegacySessionToken","getAuthForSessionToken","UNKNOWN_ERROR","authorization","header","authPrefix","match","toLowerCase","encodedAuth","substring","credentials","decodeBase64","jsKeyPrefix","matchKey","str","allowCrossDomain","allowHeaders","join","baseOrigins","allowOrigin","requestOrigin","origin","allowOrigins","method","sendStatus","allowMethodOverride","_method","originalMethod","handleParseErrors","enableExpressErrorHandler","httpStatus","OBJECT_NOT_FOUND","process","env","TESTING","stack","enforceMasterKeyAccess","end","promiseEnforceMasterKeyAccess","request","resolve","addRateLimit","route","cloud","RateLimitOptions","redisStore","connectionPromise","redisUrl","client","createClient","isOpen","connect","_config","RedisStore","sendCommand","args","transformPath","requestPath","push","pathToRegexp","rateLimit","windowMs","requestTimeWindow","max","requestCount","errorResponseMessage","response","options","skip","_request$auth","includeInternalRequests","includeMasterKey","requestMethods","Array","isArray","regExp","keyGenerator","zone","Server","RateLimitZone","global","token","session","_request$auth2","id","put","promiseEnsureIdempotency","database","adapter","MongoStorageAdapter","PostgresStorageAdapter","requestId","paths","ttl","idempotencyOptions","reqPath","replace","regex","charAt","expiryDate","Date","setSeconds","getSeconds","rest","create","master","reqId","expire","_encode","catch","DUPLICATE_VALUE","DUPLICATE_REQUEST","INVALID_JSON"],"sources":["../src/middlewares.js"],"sourcesContent":["import AppCache from './cache';\nimport Parse from 'parse/node';\nimport auth from './Auth';\nimport Config from './Config';\nimport ClientSDK from './ClientSDK';\nimport defaultLogger from './logger';\nimport rest from './rest';\nimport MongoStorageAdapter from './Adapters/Storage/Mongo/MongoStorageAdapter';\nimport PostgresStorageAdapter from './Adapters/Storage/Postgres/PostgresStorageAdapter';\nimport rateLimit from 'express-rate-limit';\nimport { RateLimitOptions } from './Options/Definitions';\nimport { pathToRegexp } from 'path-to-regexp';\nimport RedisStore from 'rate-limit-redis';\nimport { createClient } from 'redis';\nimport { BlockList, isIPv4 } from 'net';\n\nexport const DEFAULT_ALLOWED_HEADERS =\n  'X-Parse-Master-Key, X-Parse-REST-API-Key, X-Parse-Javascript-Key, X-Parse-Application-Id, X-Parse-Client-Version, X-Parse-Session-Token, X-Requested-With, X-Parse-Revocable-Session, X-Parse-Request-Id, Content-Type, Pragma, Cache-Control';\n\nconst getMountForRequest = function (req) {\n  const mountPathLength = req.originalUrl.length - req.url.length;\n  const mountPath = req.originalUrl.slice(0, mountPathLength);\n  return req.protocol + '://' + req.get('host') + mountPath;\n};\n\nconst getBlockList = (ipRangeList, store) => {\n  if (store.get('blockList')) return store.get('blockList');\n  const blockList = new BlockList();\n  ipRangeList.forEach(fullIp => {\n    if (fullIp === '::/0' || fullIp === '::') {\n      store.set('allowAllIpv6', true);\n      return;\n    }\n    if (fullIp === '0.0.0.0/0' || fullIp === '0.0.0.0') {\n      store.set('allowAllIpv4', true);\n      return;\n    }\n    const [ip, mask] = fullIp.split('/');\n    if (!mask) {\n      blockList.addAddress(ip, isIPv4(ip) ? 'ipv4' : 'ipv6');\n    } else {\n      blockList.addSubnet(ip, Number(mask), isIPv4(ip) ? 'ipv4' : 'ipv6');\n    }\n  });\n  store.set('blockList', blockList);\n  return blockList;\n};\n\nexport const checkIp = (ip, ipRangeList, store) => {\n  const incomingIpIsV4 = isIPv4(ip);\n  const blockList = getBlockList(ipRangeList, store);\n\n  if (store.get(ip)) return true;\n  if (store.get('allowAllIpv4') && incomingIpIsV4) return true;\n  if (store.get('allowAllIpv6') && !incomingIpIsV4) return true;\n  const result = blockList.check(ip, incomingIpIsV4 ? 'ipv4' : 'ipv6');\n\n  // If the ip is in the list, we store the result in the store\n  // so we have a optimized path for the next request\n  if (ipRangeList.includes(ip) && result) {\n    store.set(ip, result);\n  }\n  return result;\n};\n\n// Checks that the request is authorized for this app and checks user\n// auth too.\n// The bodyparser should run before this middleware.\n// Adds info to the request:\n// req.config - the Config for this app\n// req.auth - the Auth for this request\nexport function handleParseHeaders(req, res, next) {\n  var mount = getMountForRequest(req);\n\n  let context = {};\n  if (req.get('X-Parse-Cloud-Context') != null) {\n    try {\n      context = JSON.parse(req.get('X-Parse-Cloud-Context'));\n      if (Object.prototype.toString.call(context) !== '[object Object]') {\n        throw 'Context is not an object';\n      }\n    } catch (e) {\n      return malformedContext(req, res);\n    }\n  }\n  var info = {\n    appId: req.get('X-Parse-Application-Id'),\n    sessionToken: req.get('X-Parse-Session-Token'),\n    masterKey: req.get('X-Parse-Master-Key'),\n    maintenanceKey: req.get('X-Parse-Maintenance-Key'),\n    installationId: req.get('X-Parse-Installation-Id'),\n    clientKey: req.get('X-Parse-Client-Key'),\n    javascriptKey: req.get('X-Parse-Javascript-Key'),\n    dotNetKey: req.get('X-Parse-Windows-Key'),\n    restAPIKey: req.get('X-Parse-REST-API-Key'),\n    clientVersion: req.get('X-Parse-Client-Version'),\n    context: context,\n  };\n\n  var basicAuth = httpAuth(req);\n\n  if (basicAuth) {\n    var basicAuthAppId = basicAuth.appId;\n    if (AppCache.get(basicAuthAppId)) {\n      info.appId = basicAuthAppId;\n      info.masterKey = basicAuth.masterKey || info.masterKey;\n      info.javascriptKey = basicAuth.javascriptKey || info.javascriptKey;\n    }\n  }\n\n  if (req.body) {\n    // Unity SDK sends a _noBody key which needs to be removed.\n    // Unclear at this point if action needs to be taken.\n    delete req.body._noBody;\n  }\n\n  var fileViaJSON = false;\n\n  if (!info.appId || !AppCache.get(info.appId)) {\n    // See if we can find the app id on the body.\n    if (req.body instanceof Buffer) {\n      // The only chance to find the app id is if this is a file\n      // upload that actually is a JSON body. So try to parse it.\n      // https://github.com/parse-community/parse-server/issues/6589\n      // It is also possible that the client is trying to upload a file but forgot\n      // to provide x-parse-app-id in header and parse a binary file will fail\n      try {\n        req.body = JSON.parse(req.body);\n      } catch (e) {\n        return invalidRequest(req, res);\n      }\n      fileViaJSON = true;\n    }\n\n    if (req.body) {\n      delete req.body._RevocableSession;\n    }\n\n    if (\n      req.body &&\n      req.body._ApplicationId &&\n      AppCache.get(req.body._ApplicationId) &&\n      (!info.masterKey || AppCache.get(req.body._ApplicationId).masterKey === info.masterKey)\n    ) {\n      info.appId = req.body._ApplicationId;\n      info.javascriptKey = req.body._JavaScriptKey || '';\n      delete req.body._ApplicationId;\n      delete req.body._JavaScriptKey;\n      // TODO: test that the REST API formats generated by the other\n      // SDKs are handled ok\n      if (req.body._ClientVersion) {\n        info.clientVersion = req.body._ClientVersion;\n        delete req.body._ClientVersion;\n      }\n      if (req.body._InstallationId) {\n        info.installationId = req.body._InstallationId;\n        delete req.body._InstallationId;\n      }\n      if (req.body._SessionToken) {\n        info.sessionToken = req.body._SessionToken;\n        delete req.body._SessionToken;\n      }\n      if (req.body._MasterKey) {\n        info.masterKey = req.body._MasterKey;\n        delete req.body._MasterKey;\n      }\n      if (req.body._context) {\n        if (req.body._context instanceof Object) {\n          info.context = req.body._context;\n        } else {\n          try {\n            info.context = JSON.parse(req.body._context);\n            if (Object.prototype.toString.call(info.context) !== '[object Object]') {\n              throw 'Context is not an object';\n            }\n          } catch (e) {\n            return malformedContext(req, res);\n          }\n        }\n        delete req.body._context;\n      }\n      if (req.body._ContentType) {\n        req.headers['content-type'] = req.body._ContentType;\n        delete req.body._ContentType;\n      }\n    } else {\n      return invalidRequest(req, res);\n    }\n  }\n\n  if (info.sessionToken && typeof info.sessionToken !== 'string') {\n    info.sessionToken = info.sessionToken.toString();\n  }\n\n  if (info.clientVersion) {\n    info.clientSDK = ClientSDK.fromString(info.clientVersion);\n  }\n\n  if (fileViaJSON) {\n    req.fileData = req.body.fileData;\n    // We need to repopulate req.body with a buffer\n    var base64 = req.body.base64;\n    req.body = Buffer.from(base64, 'base64');\n  }\n\n  const clientIp = getClientIp(req);\n  const config = Config.get(info.appId, mount);\n  if (config.state && config.state !== 'ok') {\n    res.status(500);\n    res.json({\n      code: Parse.Error.INTERNAL_SERVER_ERROR,\n      error: `Invalid server state: ${config.state}`,\n    });\n    return;\n  }\n\n  info.app = AppCache.get(info.appId);\n  req.config = config;\n  req.config.headers = req.headers || {};\n  req.config.ip = clientIp;\n  req.info = info;\n\n  const isMaintenance =\n    req.config.maintenanceKey && info.maintenanceKey === req.config.maintenanceKey;\n  if (isMaintenance) {\n    if (checkIp(clientIp, req.config.maintenanceKeyIps || [], req.config.maintenanceKeyIpsStore)) {\n      req.auth = new auth.Auth({\n        config: req.config,\n        installationId: info.installationId,\n        isMaintenance: true,\n      });\n      next();\n      return;\n    }\n    const log = req.config?.loggerController || defaultLogger;\n    log.error(\n      `Request using maintenance key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'maintenanceKeyIps'.`\n    );\n  }\n\n  let isMaster = info.masterKey === req.config.masterKey;\n\n  if (isMaster && !checkIp(clientIp, req.config.masterKeyIps || [], req.config.masterKeyIpsStore)) {\n    const log = req.config?.loggerController || defaultLogger;\n    log.error(\n      `Request using master key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'masterKeyIps'.`\n    );\n    isMaster = false;\n    const error = new Error();\n    error.status = 403;\n    error.message = `unauthorized`;\n    throw error;\n  }\n\n  if (isMaster) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: true,\n    });\n    return handleRateLimit(req, res, next);\n  }\n\n  var isReadOnlyMaster = info.masterKey === req.config.readOnlyMasterKey;\n  if (\n    typeof req.config.readOnlyMasterKey != 'undefined' &&\n    req.config.readOnlyMasterKey &&\n    isReadOnlyMaster\n  ) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: true,\n      isReadOnly: true,\n    });\n    return handleRateLimit(req, res, next);\n  }\n\n  // Client keys are not required in parse-server, but if any have been configured in the server, validate them\n  //  to preserve original behavior.\n  const keys = ['clientKey', 'javascriptKey', 'dotNetKey', 'restAPIKey'];\n  const oneKeyConfigured = keys.some(function (key) {\n    return req.config[key] !== undefined;\n  });\n  const oneKeyMatches = keys.some(function (key) {\n    return req.config[key] !== undefined && info[key] === req.config[key];\n  });\n\n  if (oneKeyConfigured && !oneKeyMatches) {\n    return invalidRequest(req, res);\n  }\n\n  if (req.url == '/login') {\n    delete info.sessionToken;\n  }\n\n  if (req.userFromJWT) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: false,\n      user: req.userFromJWT,\n    });\n    return handleRateLimit(req, res, next);\n  }\n\n  if (!info.sessionToken) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: false,\n    });\n  }\n  handleRateLimit(req, res, next);\n}\n\nconst handleRateLimit = async (req, res, next) => {\n  const rateLimits = req.config.rateLimits || [];\n  try {\n    await Promise.all(\n      rateLimits.map(async limit => {\n        const pathExp = new RegExp(limit.path);\n        if (pathExp.test(req.url)) {\n          await limit.handler(req, res, err => {\n            if (err) {\n              if (err.code === Parse.Error.CONNECTION_FAILED) {\n                throw err;\n              }\n              req.config.loggerController.error(\n                'An unknown error occured when attempting to apply the rate limiter: ',\n                err\n              );\n            }\n          });\n        }\n      })\n    );\n  } catch (error) {\n    res.status(429);\n    res.json({ code: Parse.Error.CONNECTION_FAILED, error: error.message });\n    return;\n  }\n  next();\n};\n\nexport const handleParseSession = async (req, res, next) => {\n  try {\n    const info = req.info;\n    if (req.auth || req.url === '/sessions/me') {\n      next();\n      return;\n    }\n    let requestAuth = null;\n    if (\n      info.sessionToken &&\n      req.url === '/upgradeToRevocableSession' &&\n      info.sessionToken.indexOf('r:') != 0\n    ) {\n      requestAuth = await auth.getAuthForLegacySessionToken({\n        config: req.config,\n        installationId: info.installationId,\n        sessionToken: info.sessionToken,\n      });\n    } else {\n      requestAuth = await auth.getAuthForSessionToken({\n        config: req.config,\n        installationId: info.installationId,\n        sessionToken: info.sessionToken,\n      });\n    }\n    req.auth = requestAuth;\n    next();\n  } catch (error) {\n    if (error instanceof Parse.Error) {\n      next(error);\n      return;\n    }\n    // TODO: Determine the correct error scenario.\n    req.config.loggerController.error('error getting auth for sessionToken', error);\n    throw new Parse.Error(Parse.Error.UNKNOWN_ERROR, error);\n  }\n};\n\nfunction getClientIp(req) {\n  return req.ip;\n}\n\nfunction httpAuth(req) {\n  if (!(req.req || req).headers.authorization) return;\n\n  var header = (req.req || req).headers.authorization;\n  var appId, masterKey, javascriptKey;\n\n  // parse header\n  var authPrefix = 'basic ';\n\n  var match = header.toLowerCase().indexOf(authPrefix);\n\n  if (match == 0) {\n    var encodedAuth = header.substring(authPrefix.length, header.length);\n    var credentials = decodeBase64(encodedAuth).split(':');\n\n    if (credentials.length == 2) {\n      appId = credentials[0];\n      var key = credentials[1];\n\n      var jsKeyPrefix = 'javascript-key=';\n\n      var matchKey = key.indexOf(jsKeyPrefix);\n      if (matchKey == 0) {\n        javascriptKey = key.substring(jsKeyPrefix.length, key.length);\n      } else {\n        masterKey = key;\n      }\n    }\n  }\n\n  return { appId: appId, masterKey: masterKey, javascriptKey: javascriptKey };\n}\n\nfunction decodeBase64(str) {\n  return Buffer.from(str, 'base64').toString();\n}\n\nexport function allowCrossDomain(appId) {\n  return (req, res, next) => {\n    const config = Config.get(appId, getMountForRequest(req));\n    let allowHeaders = DEFAULT_ALLOWED_HEADERS;\n    if (config && config.allowHeaders) {\n      allowHeaders += `, ${config.allowHeaders.join(', ')}`;\n    }\n\n    const baseOrigins =\n      typeof config?.allowOrigin === 'string' ? [config.allowOrigin] : config?.allowOrigin ?? ['*'];\n    const requestOrigin = req.headers.origin;\n    const allowOrigins =\n      requestOrigin && baseOrigins.includes(requestOrigin) ? requestOrigin : baseOrigins[0];\n    res.header('Access-Control-Allow-Origin', allowOrigins);\n    res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');\n    res.header('Access-Control-Allow-Headers', allowHeaders);\n    res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id');\n    // intercept OPTIONS method\n    if ('OPTIONS' == req.method) {\n      res.sendStatus(200);\n    } else {\n      next();\n    }\n  };\n}\n\nexport function allowMethodOverride(req, res, next) {\n  if (req.method === 'POST' && req.body._method) {\n    req.originalMethod = req.method;\n    req.method = req.body._method;\n    delete req.body._method;\n  }\n  next();\n}\n\nexport function handleParseErrors(err, req, res, next) {\n  const log = (req.config && req.config.loggerController) || defaultLogger;\n  if (err instanceof Parse.Error) {\n    if (req.config && req.config.enableExpressErrorHandler) {\n      return next(err);\n    }\n    let httpStatus;\n    // TODO: fill out this mapping\n    switch (err.code) {\n      case Parse.Error.INTERNAL_SERVER_ERROR:\n        httpStatus = 500;\n        break;\n      case Parse.Error.OBJECT_NOT_FOUND:\n        httpStatus = 404;\n        break;\n      default:\n        httpStatus = 400;\n    }\n    res.status(httpStatus);\n    res.json({ code: err.code, error: err.message });\n    log.error('Parse error: ', err);\n  } else if (err.status && err.message) {\n    res.status(err.status);\n    res.json({ error: err.message });\n    if (!(process && process.env.TESTING)) {\n      next(err);\n    }\n  } else {\n    log.error('Uncaught internal server error.', err, err.stack);\n    res.status(500);\n    res.json({\n      code: Parse.Error.INTERNAL_SERVER_ERROR,\n      message: 'Internal server error.',\n    });\n    if (!(process && process.env.TESTING)) {\n      next(err);\n    }\n  }\n}\n\nexport function enforceMasterKeyAccess(req, res, next) {\n  if (!req.auth.isMaster) {\n    res.status(403);\n    res.end('{\"error\":\"unauthorized: master key is required\"}');\n    return;\n  }\n  next();\n}\n\nexport function promiseEnforceMasterKeyAccess(request) {\n  if (!request.auth.isMaster) {\n    const error = new Error();\n    error.status = 403;\n    error.message = 'unauthorized: master key is required';\n    throw error;\n  }\n  return Promise.resolve();\n}\n\nexport const addRateLimit = (route, config, cloud) => {\n  if (typeof config === 'string') {\n    config = Config.get(config);\n  }\n  for (const key in route) {\n    if (!RateLimitOptions[key]) {\n      throw `Invalid rate limit option \"${key}\"`;\n    }\n  }\n  if (!config.rateLimits) {\n    config.rateLimits = [];\n  }\n  const redisStore = {\n    connectionPromise: Promise.resolve(),\n    store: null,\n  };\n  if (route.redisUrl) {\n    const client = createClient({\n      url: route.redisUrl,\n    });\n    redisStore.connectionPromise = async () => {\n      if (client.isOpen) {\n        return;\n      }\n      try {\n        await client.connect();\n      } catch (e) {\n        const log = config?.loggerController || defaultLogger;\n        log.error(`Could not connect to redisURL in rate limit: ${e}`);\n      }\n    };\n    redisStore.connectionPromise();\n    redisStore.store = new RedisStore({\n      sendCommand: async (...args) => {\n        await redisStore.connectionPromise();\n        return client.sendCommand(args);\n      },\n    });\n  }\n  let transformPath = route.requestPath.split('/*').join('/(.*)');\n  if (transformPath === '*') {\n    transformPath = '(.*)';\n  }\n  config.rateLimits.push({\n    path: pathToRegexp(transformPath),\n    handler: rateLimit({\n      windowMs: route.requestTimeWindow,\n      max: route.requestCount,\n      message: route.errorResponseMessage || RateLimitOptions.errorResponseMessage.default,\n      handler: (request, response, next, options) => {\n        throw {\n          code: Parse.Error.CONNECTION_FAILED,\n          message: options.message,\n        };\n      },\n      skip: request => {\n        if (request.ip === '127.0.0.1' && !route.includeInternalRequests) {\n          return true;\n        }\n        if (route.includeMasterKey) {\n          return false;\n        }\n        if (route.requestMethods) {\n          if (Array.isArray(route.requestMethods)) {\n            if (!route.requestMethods.includes(request.method)) {\n              return true;\n            }\n          } else {\n            const regExp = new RegExp(route.requestMethods);\n            if (!regExp.test(request.method)) {\n              return true;\n            }\n          }\n        }\n        return request.auth?.isMaster;\n      },\n      keyGenerator: async request => {\n        if (route.zone === Parse.Server.RateLimitZone.global) {\n          return request.config.appId;\n        }\n        const token = request.info.sessionToken;\n        if (route.zone === Parse.Server.RateLimitZone.session && token) {\n          return token;\n        }\n        if (route.zone === Parse.Server.RateLimitZone.user && token) {\n          if (!request.auth) {\n            await new Promise(resolve => handleParseSession(request, null, resolve));\n          }\n          if (request.auth?.user?.id && request.zone === 'user') {\n            return request.auth.user.id;\n          }\n        }\n        return request.config.ip;\n      },\n      store: redisStore.store,\n    }),\n    cloud,\n  });\n  Config.put(config);\n};\n\n/**\n * Deduplicates a request to ensure idempotency. Duplicates are determined by the request ID\n * in the request header. If a request has no request ID, it is executed anyway.\n * @param {*} req The request to evaluate.\n * @returns Promise<{}>\n */\nexport function promiseEnsureIdempotency(req) {\n  // Enable feature only for MongoDB\n  if (\n    !(\n      req.config.database.adapter instanceof MongoStorageAdapter ||\n      req.config.database.adapter instanceof PostgresStorageAdapter\n    )\n  ) {\n    return Promise.resolve();\n  }\n  // Get parameters\n  const config = req.config;\n  const requestId = ((req || {}).headers || {})['x-parse-request-id'];\n  const { paths, ttl } = config.idempotencyOptions;\n  if (!requestId || !config.idempotencyOptions) {\n    return Promise.resolve();\n  }\n  // Request path may contain trailing slashes, depending on the original request, so remove\n  // leading and trailing slashes to make it easier to specify paths in the configuration\n  const reqPath = req.path.replace(/^\\/|\\/$/, '');\n  // Determine whether idempotency is enabled for current request path\n  let match = false;\n  for (const path of paths) {\n    // Assume one wants a path to always match from the beginning to prevent any mistakes\n    const regex = new RegExp(path.charAt(0) === '^' ? path : '^' + path);\n    if (reqPath.match(regex)) {\n      match = true;\n      break;\n    }\n  }\n  if (!match) {\n    return Promise.resolve();\n  }\n  // Try to store request\n  const expiryDate = new Date(new Date().setSeconds(new Date().getSeconds() + ttl));\n  return rest\n    .create(config, auth.master(config), '_Idempotency', {\n      reqId: requestId,\n      expire: Parse._encode(expiryDate),\n    })\n    .catch(e => {\n      if (e.code == Parse.Error.DUPLICATE_VALUE) {\n        throw new Parse.Error(Parse.Error.DUPLICATE_REQUEST, 'Duplicate request');\n      }\n      throw e;\n    });\n}\n\nfunction invalidRequest(req, res) {\n  res.status(403);\n  res.end('{\"error\":\"unauthorized\"}');\n}\n\nfunction malformedContext(req, res) {\n  res.status(400);\n  res.json({ code: Parse.Error.INVALID_JSON, error: 'Invalid object for context.' });\n}\n"],"mappings":"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","ignoreList":[]}