parse-server 9.7.0 → 9.7.1-alpha.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/Routers/FilesRouter.js +34 -6
- package/lib/batch.js +4 -1
- package/package.json +7 -7
|
@@ -16,6 +16,7 @@ function _interopRequireWildcard(e, t) { if ("function" == typeof WeakMap) var r
|
|
|
16
16
|
function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
|
|
17
17
|
const triggers = require('../triggers');
|
|
18
18
|
const Utils = require('../Utils');
|
|
19
|
+
const auth = require('../Auth');
|
|
19
20
|
/**
|
|
20
21
|
* Wraps a readable stream in a Readable that enforces a byte size limit.
|
|
21
22
|
* Data flow is lazy: the source is not read until a consumer starts reading
|
|
@@ -100,6 +101,21 @@ class FilesRouter {
|
|
|
100
101
|
const parts = req.params.filepath;
|
|
101
102
|
return Array.isArray(parts) ? parts.join('/') : parts;
|
|
102
103
|
}
|
|
104
|
+
static async _resolveAuth(req, config) {
|
|
105
|
+
const sessionToken = req.get('X-Parse-Session-Token');
|
|
106
|
+
if (!sessionToken) {
|
|
107
|
+
return null;
|
|
108
|
+
}
|
|
109
|
+
try {
|
|
110
|
+
return await auth.getAuthForSessionToken({
|
|
111
|
+
config,
|
|
112
|
+
sessionToken,
|
|
113
|
+
installationId: req.get('X-Parse-Installation-Id')
|
|
114
|
+
});
|
|
115
|
+
} catch {
|
|
116
|
+
return null;
|
|
117
|
+
}
|
|
118
|
+
}
|
|
103
119
|
static validateDirectory(directory) {
|
|
104
120
|
if (typeof directory !== 'string') {
|
|
105
121
|
return new _node.default.Error(_node.default.Error.INVALID_FILE_NAME, 'Directory must be a string.');
|
|
@@ -147,9 +163,10 @@ class FilesRouter {
|
|
|
147
163
|
let file = new _node.default.File(filename, {
|
|
148
164
|
base64: ''
|
|
149
165
|
}, contentType);
|
|
166
|
+
const fileAuth = await FilesRouter._resolveAuth(req, config);
|
|
150
167
|
const triggerResult = await triggers.maybeRunFileTrigger(triggers.Types.beforeFind, {
|
|
151
168
|
file
|
|
152
|
-
}, config,
|
|
169
|
+
}, config, fileAuth);
|
|
153
170
|
if (triggerResult?.file?._name) {
|
|
154
171
|
filename = triggerResult?.file?._name;
|
|
155
172
|
contentType = mime.getType(filename);
|
|
@@ -158,7 +175,17 @@ class FilesRouter {
|
|
|
158
175
|
'X-Content-Type-Options': 'nosniff'
|
|
159
176
|
};
|
|
160
177
|
if (isFileStreamable(req, filesController)) {
|
|
161
|
-
|
|
178
|
+
const afterFind = await triggers.maybeRunFileTrigger(triggers.Types.afterFind, {
|
|
179
|
+
file,
|
|
180
|
+
forceDownload: false,
|
|
181
|
+
responseHeaders: {
|
|
182
|
+
...defaultResponseHeaders
|
|
183
|
+
}
|
|
184
|
+
}, config, fileAuth);
|
|
185
|
+
if (afterFind?.forceDownload) {
|
|
186
|
+
res.set('Content-Disposition', `attachment;filename=${afterFind.file?._name || filename}`);
|
|
187
|
+
}
|
|
188
|
+
for (const [key, value] of Object.entries(afterFind?.responseHeaders ?? defaultResponseHeaders)) {
|
|
162
189
|
res.set(key, value);
|
|
163
190
|
}
|
|
164
191
|
filesController.handleFileStream(config, filename, req, res, contentType).catch(() => {
|
|
@@ -185,7 +212,7 @@ class FilesRouter {
|
|
|
185
212
|
responseHeaders: {
|
|
186
213
|
...defaultResponseHeaders
|
|
187
214
|
}
|
|
188
|
-
}, config,
|
|
215
|
+
}, config, fileAuth);
|
|
189
216
|
if (afterFind?.file) {
|
|
190
217
|
contentType = mime.getType(afterFind.file._name);
|
|
191
218
|
data = Buffer.from(afterFind.file._data, 'base64');
|
|
@@ -676,9 +703,10 @@ class FilesRouter {
|
|
|
676
703
|
const file = new _node.default.File(filename, {
|
|
677
704
|
base64: ''
|
|
678
705
|
});
|
|
706
|
+
const fileAuth = await FilesRouter._resolveAuth(req, config);
|
|
679
707
|
const triggerResult = await triggers.maybeRunFileTrigger(triggers.Types.beforeFind, {
|
|
680
708
|
file
|
|
681
|
-
}, config,
|
|
709
|
+
}, config, fileAuth);
|
|
682
710
|
if (triggerResult?.file?._name) {
|
|
683
711
|
filename = triggerResult.file._name;
|
|
684
712
|
}
|
|
@@ -691,7 +719,7 @@ class FilesRouter {
|
|
|
691
719
|
}
|
|
692
720
|
await triggers.maybeRunFileTrigger(triggers.Types.afterFind, {
|
|
693
721
|
file
|
|
694
|
-
}, config,
|
|
722
|
+
}, config, fileAuth);
|
|
695
723
|
res.status(200);
|
|
696
724
|
res.json(data);
|
|
697
725
|
} catch (e) {
|
|
@@ -714,4 +742,4 @@ function isFileStreamable(req, filesController) {
|
|
|
714
742
|
const end = Number(range[1]);
|
|
715
743
|
return (!isNaN(start) || !isNaN(end)) && typeof filesController.adapter.handleFileStream === 'function';
|
|
716
744
|
}
|
|
717
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_express","_interopRequireDefault","require","Middlewares","_interopRequireWildcard","_node","_Config","_logger","_stream","_Error","e","t","WeakMap","r","n","__esModule","o","i","f","__proto__","default","has","get","set","hasOwnProperty","call","Object","defineProperty","getOwnPropertyDescriptor","triggers","Utils","createSizeLimitedStream","source","maxBytes","totalBytes","started","sourceEnded","onData","onEnd","onError","output","Readable","read","chunk","length","destroy","Parse","Error","FILE_SAVE_ERROR","push","pause","err","on","resume","callback","removeListener","RESERVED_DIRECTORY_SEGMENTS","exports","FilesRouter","expressRouter","maxUploadSize","router","express","Router","metadataHandler","getHandler","post","req","res","next","INVALID_FILE_NAME","_earlyHeadersMiddleware","_bodyParsingMiddleware","handleParseHeaders","handleParseSession","createHandler","bind","delete","enforceMasterKeyAccess","deleteHandler","_getFilenameFromParams","parts","params","filepath","Array","isArray","join","validateDirectory","directory","includes","startsWith","endsWith","firstSegment","split","dirRegex","test","config","Config","appId","error","createSanitizedHttpError","status","json","message","filename","filesController","mime","contentType","getType","file","File","base64","triggerResult","maybeRunFileTrigger","Types","beforeFind","auth","_name","defaultResponseHeaders","isFileStreamable","key","value","entries","handleFileStream","catch","end","data","getFileData","toString","afterFind","forceDownload","responseHeaders","Buffer","from","_data","resolveError","code","SCRIPT_FAILED","maxUploadSizeOverride","undefined","masterKey","loadMasterKey","masterKeyIps","checkIp","ip","masterKeyIpsStore","parsedBytes","parseSizeToBytes","_maxUploadSizeOverride","defaultMaxBytes","_maxUploadSizeBytes","limit","raw","type","isReadOnly","user","isMaster","isLinked","AnonymousUtils","fileUpload","enableForAnonymousUser","enableForAuthenticatedUser","enableForPublic","validateFilename","fileExtensions","isValidExtension","extension","some","ext","regex","RegExp","substring","lastIndexOf","replace","fileData","parsed","JSON","parse","metadata","INVALID_JSON","tags","OPERATION_FORBIDDEN","directoryError","isBuffer","body","_handleBufferedUpload","_handleStreamUpload","checkProhibitedKeywords","INVALID_KEY_NAME","setTags","setMetadata","setDirectory","fileSize","byteLength","fileObject","beforeSave","saveResult","url","name","bufferData","_source","format","buffer","fileOptions","_metadata","fileTags","keys","_tags","assign","_directory","createFileResult","createFile","_url","_requestTask","_previousSave","Promise","resolve","afterSave","logger","stream","contentLength","parseInt","hasExtension","sourceType","adapter","getFileLocation","beforeDelete","deleteFile","afterDelete","FILE_DELETE_ERROR","getMetadata","range","start","Number","isNaN"],"sources":["../../src/Routers/FilesRouter.js"],"sourcesContent":["import express from 'express';\nimport * as Middlewares from '../middlewares';\nimport Parse from 'parse/node';\nimport Config from '../Config';\nimport logger from '../logger';\nconst triggers = require('../triggers');\nconst Utils = require('../Utils');\nimport { Readable } from 'stream';\nimport { createSanitizedHttpError } from '../Error';\n\n/**\n * Wraps a readable stream in a Readable that enforces a byte size limit.\n * Data flow is lazy: the source is not read until a consumer starts reading\n * from the returned stream (via pipe or 'data' listener). This ensures the\n * consumer's error listener is attached before any data (or error) is emitted.\n */\nexport function createSizeLimitedStream(source, maxBytes) {\n  let totalBytes = 0;\n  let started = false;\n  let sourceEnded = false;\n  let onData, onEnd, onError;\n\n  const output = new Readable({\n    read() {\n      if (!started) {\n        started = true;\n\n        onData = (chunk) => {\n          totalBytes += chunk.length;\n          if (totalBytes > maxBytes) {\n            output.destroy(\n              new Parse.Error(\n                Parse.Error.FILE_SAVE_ERROR,\n                `File size exceeds maximum allowed: ${maxBytes} bytes.`\n              )\n            );\n            return;\n          }\n          if (!output.push(chunk)) {\n            source.pause();\n          }\n        };\n\n        onEnd = () => {\n          sourceEnded = true;\n          output.push(null);\n        };\n\n        onError = (err) => output.destroy(err);\n\n        source.on('data', onData);\n        source.on('end', onEnd);\n        source.on('error', onError);\n      }\n\n      // Resume source in case it was paused due to backpressure\n      if (!sourceEnded) {\n        source.resume();\n      }\n    },\n    destroy(err, callback) {\n      if (onData) {\n        source.removeListener('data', onData);\n      }\n      if (onEnd) {\n        source.removeListener('end', onEnd);\n      }\n      if (onError) {\n        source.removeListener('error', onError);\n      }\n      // Suppress errors emitted during drain (e.g. client disconnect)\n      source.on('error', () => {});\n      if (!sourceEnded) {\n        source.resume();\n      }\n      callback(err);\n    }\n  });\n\n  return output;\n}\n\n// Segments that conflict with sub-routes under GET /files/:appId/*. If a file\n// directory starts with one of these, its URL would match the wrong route\n// handler. Update this list when adding new sub-routes to expressRouter().\nexport const RESERVED_DIRECTORY_SEGMENTS = ['metadata'];\n\nexport class FilesRouter {\n  expressRouter({ maxUploadSize = '20Mb' } = {}) {\n    var router = express.Router();\n    // Metadata route must come before the catch-all GET route\n    router.get('/files/:appId/metadata/*filepath', this.metadataHandler);\n    router.get('/files/:appId/*filepath', this.getHandler);\n\n    router.post('/files', function (req, res, next) {\n      next(new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Filename not provided.'));\n    });\n\n    router.post(\n      '/files/:filename',\n      this._earlyHeadersMiddleware(),\n      this._bodyParsingMiddleware(maxUploadSize),\n      Middlewares.handleParseHeaders,\n      Middlewares.handleParseSession,\n      this.createHandler.bind(this)\n    );\n\n    router.delete(\n      '/files/*filepath',\n      Middlewares.handleParseHeaders,\n      Middlewares.handleParseSession,\n      Middlewares.enforceMasterKeyAccess,\n      this.deleteHandler\n    );\n    return router;\n  }\n\n  static _getFilenameFromParams(req) {\n    const parts = req.params.filepath;\n    return Array.isArray(parts) ? parts.join('/') : parts;\n  }\n\n  static validateDirectory(directory) {\n    if (typeof directory !== 'string') {\n      return new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Directory must be a string.');\n    }\n    if (directory.length === 0) {\n      return new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Directory must not be empty.');\n    }\n    if (directory.length > 256) {\n      return new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Directory path is too long.');\n    }\n    if (directory.includes('..')) {\n      return new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Directory must not contain \"..\".');\n    }\n    if (directory.startsWith('/') || directory.endsWith('/')) {\n      return new Parse.Error(\n        Parse.Error.INVALID_FILE_NAME,\n        'Directory must not start or end with \"/\".'\n      );\n    }\n    if (directory.includes('//')) {\n      return new Parse.Error(\n        Parse.Error.INVALID_FILE_NAME,\n        'Directory must not contain consecutive slashes.'\n      );\n    }\n    const firstSegment = directory.split('/')[0];\n    if (RESERVED_DIRECTORY_SEGMENTS.includes(firstSegment)) {\n      return new Parse.Error(\n        Parse.Error.INVALID_FILE_NAME,\n        `Directory must not start with reserved segment \"${firstSegment}\".`\n      );\n    }\n    const dirRegex = /^[a-zA-Z0-9][a-zA-Z0-9_\\-/]*$/;\n    if (!dirRegex.test(directory)) {\n      return new Parse.Error(\n        Parse.Error.INVALID_FILE_NAME,\n        'Directory contains invalid characters.'\n      );\n    }\n    return null;\n  }\n\n  async getHandler(req, res) {\n    const config = Config.get(req.params.appId);\n    if (!config) {\n      const error = createSanitizedHttpError(403, 'Invalid application ID.', config);\n      res.status(error.status);\n      res.json({ error: error.message });\n      return;\n    }\n\n    let filename = FilesRouter._getFilenameFromParams(req);\n    try {\n      const filesController = config.filesController;\n      const mime = (await import('mime')).default;\n      let contentType = mime.getType(filename);\n      let file = new Parse.File(filename, { base64: '' }, contentType);\n      const triggerResult = await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeFind,\n        { file },\n        config,\n        req.auth\n      );\n      if (triggerResult?.file?._name) {\n        filename = triggerResult?.file?._name;\n        contentType = mime.getType(filename);\n      }\n\n      const defaultResponseHeaders = { 'X-Content-Type-Options': 'nosniff' };\n\n      if (isFileStreamable(req, filesController)) {\n        for (const [key, value] of Object.entries(defaultResponseHeaders)) {\n          res.set(key, value);\n        }\n        filesController.handleFileStream(config, filename, req, res, contentType).catch(() => {\n          res.status(404);\n          res.set('Content-Type', 'text/plain');\n          res.end('File not found.');\n        });\n        return;\n      }\n\n      let data = await filesController.getFileData(config, filename).catch(() => {\n        res.status(404);\n        res.set('Content-Type', 'text/plain');\n        res.end('File not found.');\n      });\n      if (!data) {\n        return;\n      }\n      file = new Parse.File(filename, { base64: data.toString('base64') }, contentType);\n      const afterFind = await triggers.maybeRunFileTrigger(\n        triggers.Types.afterFind,\n        { file, forceDownload: false, responseHeaders: { ...defaultResponseHeaders } },\n        config,\n        req.auth\n      );\n\n      if (afterFind?.file) {\n        contentType = mime.getType(afterFind.file._name);\n        data = Buffer.from(afterFind.file._data, 'base64');\n      }\n\n      res.status(200);\n      res.set('Content-Type', contentType);\n      res.set('Content-Length', data.length);\n      if (afterFind.forceDownload) {\n        res.set('Content-Disposition', `attachment;filename=${afterFind.file._name}`);\n      }\n      if (afterFind.responseHeaders) {\n        for (const [key, value] of Object.entries(afterFind.responseHeaders)) {\n          res.set(key, value);\n        }\n      }\n      res.end(data);\n    } catch (e) {\n      const err = triggers.resolveError(e, {\n        code: Parse.Error.SCRIPT_FAILED,\n        message: `Could not find file: ${filename}.`,\n      });\n      res.status(403);\n      res.json({ code: err.code, error: err.message });\n    }\n  }\n\n  /**\n   * Middleware that runs before body parsing to handle headers that must be\n   * resolved before the request body is consumed. Currently supports:\n   *\n   * - `X-Parse-File-Max-Upload-Size`: Overrides the server-wide `maxUploadSize`\n   *   for this request. Requires the master key. The value uses the same format\n   *   as the server option (e.g. `'50mb'`, `'1gb'`). Sets `req._maxUploadSizeOverride`\n   *   (in bytes) for `_bodyParsingMiddleware` to use.\n   */\n  _earlyHeadersMiddleware() {\n    return async (req, res, next) => {\n      const maxUploadSizeOverride = req.get('X-Parse-File-Max-Upload-Size');\n      if (!maxUploadSizeOverride) {\n        return next();\n      }\n      const appId = req.get('X-Parse-Application-Id');\n      const config = Config.get(appId);\n      if (!config) {\n        const error = createSanitizedHttpError(403, 'Invalid application ID.', undefined);\n        res.status(error.status);\n        res.json({ error: error.message });\n        return;\n      }\n      const masterKey = await config.loadMasterKey();\n      if (req.get('X-Parse-Master-Key') !== masterKey) {\n        const error = createSanitizedHttpError(403, 'unauthorized: master key is required', config);\n        res.status(error.status);\n        res.json({ error: error.message });\n        return;\n      }\n      if (config.masterKeyIps?.length && !Middlewares.checkIp(req.ip, config.masterKeyIps, config.masterKeyIpsStore)) {\n        const error = createSanitizedHttpError(403, 'unauthorized: master key is required', config);\n        res.status(error.status);\n        res.json({ error: error.message });\n        return;\n      }\n      let parsedBytes;\n      try {\n        parsedBytes = Utils.parseSizeToBytes(maxUploadSizeOverride);\n      } catch {\n        return next(\n          new Parse.Error(\n            Parse.Error.FILE_SAVE_ERROR,\n            `Invalid maxUploadSize override value: ${maxUploadSizeOverride}`\n          )\n        );\n      }\n      req._maxUploadSizeOverride = parsedBytes;\n      next();\n    };\n  }\n\n  _bodyParsingMiddleware(maxUploadSize) {\n    const defaultMaxBytes = Utils.parseSizeToBytes(maxUploadSize);\n    return (req, res, next) => {\n      if (req.get('X-Parse-Upload-Mode') === 'stream') {\n        req._maxUploadSizeBytes = req._maxUploadSizeOverride ?? defaultMaxBytes;\n        return next();\n      }\n      const limit = req._maxUploadSizeOverride ?? maxUploadSize;\n      return express.raw({ type: () => true, limit })(req, res, next);\n    };\n  }\n\n  async createHandler(req, res, next) {\n    if (req.auth.isReadOnly) {\n      const error = createSanitizedHttpError(403, \"read-only masterKey isn't allowed to create a file.\", req.config);\n      res.status(error.status);\n      res.end(`{\"error\":\"${error.message}\"}`);\n      return;\n    }\n    const config = req.config;\n    const user = req.auth.user;\n    const isMaster = req.auth.isMaster;\n    const isLinked = user && Parse.AnonymousUtils.isLinked(user);\n    if (!isMaster && !config.fileUpload.enableForAnonymousUser && isLinked) {\n      next(\n        new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'File upload by anonymous user is disabled.')\n      );\n      return;\n    }\n    if (!isMaster && !config.fileUpload.enableForAuthenticatedUser && !isLinked && user) {\n      next(\n        new Parse.Error(\n          Parse.Error.FILE_SAVE_ERROR,\n          'File upload by authenticated user is disabled.'\n        )\n      );\n      return;\n    }\n    if (!isMaster && !config.fileUpload.enableForPublic && !user) {\n      next(new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'File upload by public is disabled.'));\n      return;\n    }\n    const filesController = config.filesController;\n    const { filename } = req.params;\n    const contentType = req.get('Content-type');\n\n    const error = filesController.validateFilename(filename);\n    if (error) {\n      next(error);\n      return;\n    }\n\n    const fileExtensions = config.fileUpload?.fileExtensions;\n    if (!isMaster && fileExtensions) {\n      const isValidExtension = extension => {\n        return fileExtensions.some(ext => {\n          if (ext === '*') {\n            return true;\n          }\n          const regex = new RegExp(ext);\n          if (regex.test(extension)) {\n            return true;\n          }\n        });\n      };\n      let extension = contentType;\n      if (filename && filename.includes('.')) {\n        extension = filename.substring(filename.lastIndexOf('.') + 1);\n      } else if (contentType && contentType.includes('/')) {\n        extension = contentType.split('/')[1];\n      }\n      // Strip MIME parameters (e.g. \";charset=utf-8\") and whitespace\n      extension = extension?.split(';')[0]?.replace(/\\s+/g, '');\n\n      if (extension && !isValidExtension(extension)) {\n        next(\n          new Parse.Error(\n            Parse.Error.FILE_SAVE_ERROR,\n            `File upload of extension ${extension} is disabled.`\n          )\n        );\n        return;\n      }\n    }\n\n    // For streaming uploads, read file data from headers since the body is the raw stream\n    if (req.get('X-Parse-Upload-Mode') === 'stream') {\n      req.fileData = {};\n      if (req.get('X-Parse-File-Directory')) {\n        req.fileData.directory = req.get('X-Parse-File-Directory');\n      }\n      if (req.get('X-Parse-File-Metadata')) {\n        try {\n          const parsed = JSON.parse(req.get('X-Parse-File-Metadata'));\n          if (!parsed || typeof parsed !== 'object' || Array.isArray(parsed)) {\n            throw new Error();\n          }\n          req.fileData.metadata = parsed;\n        } catch {\n          next(new Parse.Error(Parse.Error.INVALID_JSON, 'Invalid JSON in X-Parse-File-Metadata header.'));\n          return;\n        }\n      }\n      if (req.get('X-Parse-File-Tags')) {\n        try {\n          const parsed = JSON.parse(req.get('X-Parse-File-Tags'));\n          if (!parsed || typeof parsed !== 'object' || Array.isArray(parsed)) {\n            throw new Error();\n          }\n          req.fileData.tags = parsed;\n        } catch {\n          next(new Parse.Error(Parse.Error.INVALID_JSON, 'Invalid JSON in X-Parse-File-Tags header.'));\n          return;\n        }\n      }\n    }\n\n    // Validate directory option (requires master key)\n    const directory = req.fileData?.directory;\n    if (directory !== undefined) {\n      if (!isMaster) {\n        next(\n          new Parse.Error(\n            Parse.Error.OPERATION_FORBIDDEN,\n            'Directory can only be set using the Master Key.'\n          )\n        );\n        return;\n      }\n      const directoryError = FilesRouter.validateDirectory(directory);\n      if (directoryError) {\n        next(directoryError);\n        return;\n      }\n    }\n\n    // Dispatch to the appropriate handler based on whether the body was buffered\n    if (Buffer.isBuffer(req.body)) {\n      return this._handleBufferedUpload(req, res, next);\n    }\n    return this._handleStreamUpload(req, res, next);\n  }\n\n  async _handleBufferedUpload(req, res, next) {\n    const config = req.config;\n    const filesController = config.filesController;\n    const { filename } = req.params;\n    const contentType = req.get('Content-type');\n\n    if (!req.body || !req.body.length) {\n      next(new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'Invalid file upload.'));\n      return;\n    }\n\n    const base64 = req.body.toString('base64');\n    const file = new Parse.File(filename, { base64 }, contentType);\n    const { metadata = {}, tags = {}, directory } = req.fileData || {};\n    try {\n      // Scan request data for denied keywords\n      Utils.checkProhibitedKeywords(config, metadata);\n      Utils.checkProhibitedKeywords(config, tags);\n    } catch (error) {\n      next(new Parse.Error(Parse.Error.INVALID_KEY_NAME, error));\n      return;\n    }\n    file.setTags(tags);\n    file.setMetadata(metadata);\n    if (directory) {\n      file.setDirectory(directory);\n    }\n    const fileSize = Buffer.byteLength(req.body);\n    const fileObject = { file, fileSize };\n    try {\n      // run beforeSaveFile trigger\n      const triggerResult = await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeSave,\n        fileObject,\n        config,\n        req.auth\n      );\n      let saveResult;\n      // if a new ParseFile is returned check if it's an already saved file\n      if (triggerResult instanceof Parse.File) {\n        fileObject.file = triggerResult;\n        if (triggerResult.url()) {\n          // set fileSize to null because we wont know how big it is here\n          fileObject.fileSize = null;\n          saveResult = {\n            url: triggerResult.url(),\n            name: triggerResult._name,\n          };\n        }\n      }\n      // if the file returned by the trigger has already been saved skip saving anything\n      if (!saveResult) {\n        // update fileSize\n        let bufferData;\n        if (fileObject.file._source?.format === 'buffer') {\n          bufferData = fileObject.file._source.buffer;\n        } else {\n          bufferData = Buffer.from(fileObject.file._data, 'base64');\n        }\n        fileObject.fileSize = Buffer.byteLength(bufferData);\n        // prepare file options\n        const fileOptions = {\n          metadata: fileObject.file._metadata,\n        };\n        // some s3-compatible providers (DigitalOcean, Linode) do not accept tags\n        // so we do not include the tags option if it is empty.\n        const fileTags =\n          Object.keys(fileObject.file._tags).length > 0 ? { tags: fileObject.file._tags } : {};\n        Object.assign(fileOptions, fileTags);\n        // include directory if set (from client request or beforeSaveFile trigger)\n        if (fileObject.file._directory) {\n          fileOptions.directory = fileObject.file._directory;\n        }\n        // save file\n        const createFileResult = await filesController.createFile(\n          config,\n          fileObject.file._name,\n          bufferData,\n          fileObject.file._source.type,\n          fileOptions\n        );\n        // update file with new data\n        fileObject.file._name = createFileResult.name;\n        fileObject.file._url = createFileResult.url;\n        fileObject.file._requestTask = null;\n        fileObject.file._previousSave = Promise.resolve(fileObject.file);\n        saveResult = {\n          url: createFileResult.url,\n          name: createFileResult.name,\n        };\n      }\n      // run afterSaveFile trigger\n      await triggers.maybeRunFileTrigger(triggers.Types.afterSave, fileObject, config, req.auth);\n      res.status(201);\n      res.set('Location', saveResult.url);\n      res.json(saveResult);\n    } catch (e) {\n      logger.error('Error creating a file: ', e);\n      const error = triggers.resolveError(e, {\n        code: Parse.Error.FILE_SAVE_ERROR,\n        message: `Could not store file: ${fileObject.file._name}.`,\n      });\n      next(error);\n    }\n  }\n\n  async _handleStreamUpload(req, res, next) {\n    const config = req.config;\n    const filesController = config.filesController;\n    const { filename } = req.params;\n    let contentType = req.get('Content-Type');\n    const maxBytes = req._maxUploadSizeBytes;\n    let stream;\n\n    try {\n      // Early rejection via Content-Length header\n      const contentLength = req.get('Content-Length');\n      if (contentLength && parseInt(contentLength, 10) > maxBytes) {\n        req.resume();\n        next(new Parse.Error(\n          Parse.Error.FILE_SAVE_ERROR,\n          `File size exceeds maximum allowed: ${maxBytes} bytes.`\n        ));\n        return;\n      }\n\n      const mime = (await import('mime')).default;\n\n      // Infer content type from extension or add extension from content type\n      const hasExtension = filename && filename.includes('.');\n      if (hasExtension && !contentType) {\n        contentType = mime.getType(filename);\n      } else if (!hasExtension && contentType) {\n        // extension will be added by filesController.createFile\n      }\n\n      // Create size-limited stream wrapping the request\n      stream = createSizeLimitedStream(req, maxBytes);\n\n      // Build a Parse.File with no _data (streaming mode)\n      const file = new Parse.File(filename, { base64: '' }, contentType);\n      const { metadata = {}, tags = {}, directory } = req.fileData || {};\n\n      // Validate metadata and tags for prohibited keywords\n      try {\n        Utils.checkProhibitedKeywords(config, metadata);\n        Utils.checkProhibitedKeywords(config, tags);\n      } catch (error) {\n        stream.destroy();\n        next(new Parse.Error(Parse.Error.INVALID_KEY_NAME, error));\n        return;\n      }\n\n      file.setTags(tags);\n      file.setMetadata(metadata);\n      if (directory) {\n        file.setDirectory(directory);\n      }\n\n      const fileSize = req.get('Content-Length')\n        ? parseInt(req.get('Content-Length'), 10)\n        : null;\n      const fileObject = { file, fileSize, stream: true };\n\n      // Run beforeSaveFile trigger\n      const triggerResult = await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeSave,\n        fileObject,\n        config,\n        req.auth\n      );\n\n      let saveResult;\n      // If a new ParseFile is returned, check if it's an already saved file\n      if (triggerResult instanceof Parse.File) {\n        fileObject.file = triggerResult;\n        if (triggerResult.url()) {\n          fileObject.fileSize = null;\n          saveResult = {\n            url: triggerResult.url(),\n            name: triggerResult._name,\n          };\n          // Destroy stream to remove listeners and drain request\n          stream.destroy();\n        }\n      }\n\n      // If the file returned by the trigger has already been saved, skip saving\n      if (!saveResult) {\n        // Prepare file options\n        const fileOptions = {\n          metadata: fileObject.file._metadata,\n        };\n        const fileTags =\n          Object.keys(fileObject.file._tags).length > 0 ? { tags: fileObject.file._tags } : {};\n        Object.assign(fileOptions, fileTags);\n        // include directory if set (from client request or beforeSaveFile trigger)\n        if (fileObject.file._directory) {\n          fileOptions.directory = fileObject.file._directory;\n        }\n\n        // Pass stream directly to filesController — it will buffer if adapter doesn't support streaming\n        const sourceType = fileObject.file._source?.type || contentType;\n        const createFileResult = await filesController.createFile(\n          config,\n          fileObject.file._name,\n          stream,\n          sourceType,\n          fileOptions\n        );\n\n        // Update file with new data\n        fileObject.file._name = createFileResult.name;\n        fileObject.file._url = createFileResult.url;\n        fileObject.file._requestTask = null;\n        fileObject.file._previousSave = Promise.resolve(fileObject.file);\n        saveResult = {\n          url: createFileResult.url,\n          name: createFileResult.name,\n        };\n      }\n\n      // Run afterSaveFile trigger\n      await triggers.maybeRunFileTrigger(triggers.Types.afterSave, fileObject, config, req.auth);\n      res.status(201);\n      res.set('Location', saveResult.url);\n      res.json(saveResult);\n    } catch (e) {\n      // Destroy stream to remove listeners and drain request, or resume directly\n      if (stream) {\n        stream.destroy();\n      } else {\n        req.resume();\n      }\n      logger.error('Error creating a file: ', e);\n      const error = triggers.resolveError(e, {\n        code: Parse.Error.FILE_SAVE_ERROR,\n        message: `Could not store file: ${filename}.`,\n      });\n      next(error);\n    }\n  }\n\n  async deleteHandler(req, res, next) {\n    if (req.auth.isReadOnly) {\n      const error = createSanitizedHttpError(403, \"read-only masterKey isn't allowed to delete a file.\", req.config);\n      res.status(error.status);\n      res.end(`{\"error\":\"${error.message}\"}`);\n      return;\n    }\n    try {\n      const { filesController } = req.config;\n      const filename = FilesRouter._getFilenameFromParams(req);\n      // run beforeDeleteFile trigger\n      const file = new Parse.File(filename);\n      file._url = await filesController.adapter.getFileLocation(req.config, filename);\n      const fileObject = { file, fileSize: null };\n      await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeDelete,\n        fileObject,\n        req.config,\n        req.auth\n      );\n      // delete file\n      await filesController.deleteFile(req.config, filename);\n      // run afterDeleteFile trigger\n      await triggers.maybeRunFileTrigger(\n        triggers.Types.afterDelete,\n        fileObject,\n        req.config,\n        req.auth\n      );\n      res.status(200);\n      // TODO: return useful JSON here?\n      res.end();\n    } catch (e) {\n      logger.error('Error deleting a file: ', e);\n      const error = triggers.resolveError(e, {\n        code: Parse.Error.FILE_DELETE_ERROR,\n        message: 'Could not delete file.',\n      });\n      next(error);\n    }\n  }\n\n  async metadataHandler(req, res) {\n    try {\n      const config = Config.get(req.params.appId);\n      if (!config) {\n        res.status(200);\n        res.json({});\n        return;\n      }\n      const { filesController } = config;\n      let filename = FilesRouter._getFilenameFromParams(req);\n      const file = new Parse.File(filename, { base64: '' });\n      const triggerResult = await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeFind,\n        { file },\n        config,\n        req.auth\n      );\n      if (triggerResult?.file?._name) {\n        filename = triggerResult.file._name;\n      }\n      const data = await filesController.getMetadata(filename).catch(() => {\n        res.status(200);\n        res.json({});\n      });\n      if (!data) {\n        return;\n      }\n      await triggers.maybeRunFileTrigger(\n        triggers.Types.afterFind,\n        { file },\n        config,\n        req.auth\n      );\n      res.status(200);\n      res.json(data);\n    } catch (e) {\n      const err = triggers.resolveError(e, {\n        code: Parse.Error.SCRIPT_FAILED,\n        message: 'Could not get file metadata.',\n      });\n      res.status(403);\n      res.json({ code: err.code, error: err.message });\n    }\n  }\n}\n\nfunction isFileStreamable(req, filesController) {\n  const range = (req.get('Range') || '/-/').split('-');\n  const start = Number(range[0]);\n  const end = Number(range[1]);\n  return (\n    (!isNaN(start) || !isNaN(end)) && typeof filesController.adapter.handleFileStream === 'function'\n  );\n}\n"],"mappings":"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","ignoreList":[]}
|
|
745
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_express","_interopRequireDefault","require","Middlewares","_interopRequireWildcard","_node","_Config","_logger","_stream","_Error","e","t","WeakMap","r","n","__esModule","o","i","f","__proto__","default","has","get","set","hasOwnProperty","call","Object","defineProperty","getOwnPropertyDescriptor","triggers","Utils","auth","createSizeLimitedStream","source","maxBytes","totalBytes","started","sourceEnded","onData","onEnd","onError","output","Readable","read","chunk","length","destroy","Parse","Error","FILE_SAVE_ERROR","push","pause","err","on","resume","callback","removeListener","RESERVED_DIRECTORY_SEGMENTS","exports","FilesRouter","expressRouter","maxUploadSize","router","express","Router","metadataHandler","getHandler","post","req","res","next","INVALID_FILE_NAME","_earlyHeadersMiddleware","_bodyParsingMiddleware","handleParseHeaders","handleParseSession","createHandler","bind","delete","enforceMasterKeyAccess","deleteHandler","_getFilenameFromParams","parts","params","filepath","Array","isArray","join","_resolveAuth","config","sessionToken","getAuthForSessionToken","installationId","validateDirectory","directory","includes","startsWith","endsWith","firstSegment","split","dirRegex","test","Config","appId","error","createSanitizedHttpError","status","json","message","filename","filesController","mime","contentType","getType","file","File","base64","fileAuth","triggerResult","maybeRunFileTrigger","Types","beforeFind","_name","defaultResponseHeaders","isFileStreamable","afterFind","forceDownload","responseHeaders","key","value","entries","handleFileStream","catch","end","data","getFileData","toString","Buffer","from","_data","resolveError","code","SCRIPT_FAILED","maxUploadSizeOverride","undefined","masterKey","loadMasterKey","masterKeyIps","checkIp","ip","masterKeyIpsStore","parsedBytes","parseSizeToBytes","_maxUploadSizeOverride","defaultMaxBytes","_maxUploadSizeBytes","limit","raw","type","isReadOnly","user","isMaster","isLinked","AnonymousUtils","fileUpload","enableForAnonymousUser","enableForAuthenticatedUser","enableForPublic","validateFilename","fileExtensions","isValidExtension","extension","some","ext","regex","RegExp","substring","lastIndexOf","replace","fileData","parsed","JSON","parse","metadata","INVALID_JSON","tags","OPERATION_FORBIDDEN","directoryError","isBuffer","body","_handleBufferedUpload","_handleStreamUpload","checkProhibitedKeywords","INVALID_KEY_NAME","setTags","setMetadata","setDirectory","fileSize","byteLength","fileObject","beforeSave","saveResult","url","name","bufferData","_source","format","buffer","fileOptions","_metadata","fileTags","keys","_tags","assign","_directory","createFileResult","createFile","_url","_requestTask","_previousSave","Promise","resolve","afterSave","logger","stream","contentLength","parseInt","hasExtension","sourceType","adapter","getFileLocation","beforeDelete","deleteFile","afterDelete","FILE_DELETE_ERROR","getMetadata","range","start","Number","isNaN"],"sources":["../../src/Routers/FilesRouter.js"],"sourcesContent":["import express from 'express';\nimport * as Middlewares from '../middlewares';\nimport Parse from 'parse/node';\nimport Config from '../Config';\nimport logger from '../logger';\nconst triggers = require('../triggers');\nconst Utils = require('../Utils');\nconst auth = require('../Auth');\nimport { Readable } from 'stream';\nimport { createSanitizedHttpError } from '../Error';\n\n/**\n * Wraps a readable stream in a Readable that enforces a byte size limit.\n * Data flow is lazy: the source is not read until a consumer starts reading\n * from the returned stream (via pipe or 'data' listener). This ensures the\n * consumer's error listener is attached before any data (or error) is emitted.\n */\nexport function createSizeLimitedStream(source, maxBytes) {\n  let totalBytes = 0;\n  let started = false;\n  let sourceEnded = false;\n  let onData, onEnd, onError;\n\n  const output = new Readable({\n    read() {\n      if (!started) {\n        started = true;\n\n        onData = (chunk) => {\n          totalBytes += chunk.length;\n          if (totalBytes > maxBytes) {\n            output.destroy(\n              new Parse.Error(\n                Parse.Error.FILE_SAVE_ERROR,\n                `File size exceeds maximum allowed: ${maxBytes} bytes.`\n              )\n            );\n            return;\n          }\n          if (!output.push(chunk)) {\n            source.pause();\n          }\n        };\n\n        onEnd = () => {\n          sourceEnded = true;\n          output.push(null);\n        };\n\n        onError = (err) => output.destroy(err);\n\n        source.on('data', onData);\n        source.on('end', onEnd);\n        source.on('error', onError);\n      }\n\n      // Resume source in case it was paused due to backpressure\n      if (!sourceEnded) {\n        source.resume();\n      }\n    },\n    destroy(err, callback) {\n      if (onData) {\n        source.removeListener('data', onData);\n      }\n      if (onEnd) {\n        source.removeListener('end', onEnd);\n      }\n      if (onError) {\n        source.removeListener('error', onError);\n      }\n      // Suppress errors emitted during drain (e.g. client disconnect)\n      source.on('error', () => {});\n      if (!sourceEnded) {\n        source.resume();\n      }\n      callback(err);\n    }\n  });\n\n  return output;\n}\n\n// Segments that conflict with sub-routes under GET /files/:appId/*. If a file\n// directory starts with one of these, its URL would match the wrong route\n// handler. Update this list when adding new sub-routes to expressRouter().\nexport const RESERVED_DIRECTORY_SEGMENTS = ['metadata'];\n\nexport class FilesRouter {\n  expressRouter({ maxUploadSize = '20Mb' } = {}) {\n    var router = express.Router();\n    // Metadata route must come before the catch-all GET route\n    router.get('/files/:appId/metadata/*filepath', this.metadataHandler);\n    router.get('/files/:appId/*filepath', this.getHandler);\n\n    router.post('/files', function (req, res, next) {\n      next(new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Filename not provided.'));\n    });\n\n    router.post(\n      '/files/:filename',\n      this._earlyHeadersMiddleware(),\n      this._bodyParsingMiddleware(maxUploadSize),\n      Middlewares.handleParseHeaders,\n      Middlewares.handleParseSession,\n      this.createHandler.bind(this)\n    );\n\n    router.delete(\n      '/files/*filepath',\n      Middlewares.handleParseHeaders,\n      Middlewares.handleParseSession,\n      Middlewares.enforceMasterKeyAccess,\n      this.deleteHandler\n    );\n    return router;\n  }\n\n  static _getFilenameFromParams(req) {\n    const parts = req.params.filepath;\n    return Array.isArray(parts) ? parts.join('/') : parts;\n  }\n\n  static async _resolveAuth(req, config) {\n    const sessionToken = req.get('X-Parse-Session-Token');\n    if (!sessionToken) {\n      return null;\n    }\n    try {\n      return await auth.getAuthForSessionToken({\n        config,\n        sessionToken,\n        installationId: req.get('X-Parse-Installation-Id'),\n      });\n    } catch {\n      return null;\n    }\n  }\n\n  static validateDirectory(directory) {\n    if (typeof directory !== 'string') {\n      return new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Directory must be a string.');\n    }\n    if (directory.length === 0) {\n      return new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Directory must not be empty.');\n    }\n    if (directory.length > 256) {\n      return new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Directory path is too long.');\n    }\n    if (directory.includes('..')) {\n      return new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Directory must not contain \"..\".');\n    }\n    if (directory.startsWith('/') || directory.endsWith('/')) {\n      return new Parse.Error(\n        Parse.Error.INVALID_FILE_NAME,\n        'Directory must not start or end with \"/\".'\n      );\n    }\n    if (directory.includes('//')) {\n      return new Parse.Error(\n        Parse.Error.INVALID_FILE_NAME,\n        'Directory must not contain consecutive slashes.'\n      );\n    }\n    const firstSegment = directory.split('/')[0];\n    if (RESERVED_DIRECTORY_SEGMENTS.includes(firstSegment)) {\n      return new Parse.Error(\n        Parse.Error.INVALID_FILE_NAME,\n        `Directory must not start with reserved segment \"${firstSegment}\".`\n      );\n    }\n    const dirRegex = /^[a-zA-Z0-9][a-zA-Z0-9_\\-/]*$/;\n    if (!dirRegex.test(directory)) {\n      return new Parse.Error(\n        Parse.Error.INVALID_FILE_NAME,\n        'Directory contains invalid characters.'\n      );\n    }\n    return null;\n  }\n\n  async getHandler(req, res) {\n    const config = Config.get(req.params.appId);\n    if (!config) {\n      const error = createSanitizedHttpError(403, 'Invalid application ID.', config);\n      res.status(error.status);\n      res.json({ error: error.message });\n      return;\n    }\n\n    let filename = FilesRouter._getFilenameFromParams(req);\n    try {\n      const filesController = config.filesController;\n      const mime = (await import('mime')).default;\n      let contentType = mime.getType(filename);\n      let file = new Parse.File(filename, { base64: '' }, contentType);\n      const fileAuth = await FilesRouter._resolveAuth(req, config);\n      const triggerResult = await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeFind,\n        { file },\n        config,\n        fileAuth\n      );\n      if (triggerResult?.file?._name) {\n        filename = triggerResult?.file?._name;\n        contentType = mime.getType(filename);\n      }\n\n      const defaultResponseHeaders = { 'X-Content-Type-Options': 'nosniff' };\n\n      if (isFileStreamable(req, filesController)) {\n        const afterFind = await triggers.maybeRunFileTrigger(\n          triggers.Types.afterFind,\n          { file, forceDownload: false, responseHeaders: { ...defaultResponseHeaders } },\n          config,\n          fileAuth\n        );\n        if (afterFind?.forceDownload) {\n          res.set('Content-Disposition', `attachment;filename=${afterFind.file?._name || filename}`);\n        }\n        for (const [key, value] of Object.entries(afterFind?.responseHeaders ?? defaultResponseHeaders)) {\n          res.set(key, value);\n        }\n        filesController.handleFileStream(config, filename, req, res, contentType).catch(() => {\n          res.status(404);\n          res.set('Content-Type', 'text/plain');\n          res.end('File not found.');\n        });\n        return;\n      }\n\n      let data = await filesController.getFileData(config, filename).catch(() => {\n        res.status(404);\n        res.set('Content-Type', 'text/plain');\n        res.end('File not found.');\n      });\n      if (!data) {\n        return;\n      }\n      file = new Parse.File(filename, { base64: data.toString('base64') }, contentType);\n      const afterFind = await triggers.maybeRunFileTrigger(\n        triggers.Types.afterFind,\n        { file, forceDownload: false, responseHeaders: { ...defaultResponseHeaders } },\n        config,\n        fileAuth\n      );\n\n      if (afterFind?.file) {\n        contentType = mime.getType(afterFind.file._name);\n        data = Buffer.from(afterFind.file._data, 'base64');\n      }\n\n      res.status(200);\n      res.set('Content-Type', contentType);\n      res.set('Content-Length', data.length);\n      if (afterFind.forceDownload) {\n        res.set('Content-Disposition', `attachment;filename=${afterFind.file._name}`);\n      }\n      if (afterFind.responseHeaders) {\n        for (const [key, value] of Object.entries(afterFind.responseHeaders)) {\n          res.set(key, value);\n        }\n      }\n      res.end(data);\n    } catch (e) {\n      const err = triggers.resolveError(e, {\n        code: Parse.Error.SCRIPT_FAILED,\n        message: `Could not find file: ${filename}.`,\n      });\n      res.status(403);\n      res.json({ code: err.code, error: err.message });\n    }\n  }\n\n  /**\n   * Middleware that runs before body parsing to handle headers that must be\n   * resolved before the request body is consumed. Currently supports:\n   *\n   * - `X-Parse-File-Max-Upload-Size`: Overrides the server-wide `maxUploadSize`\n   *   for this request. Requires the master key. The value uses the same format\n   *   as the server option (e.g. `'50mb'`, `'1gb'`). Sets `req._maxUploadSizeOverride`\n   *   (in bytes) for `_bodyParsingMiddleware` to use.\n   */\n  _earlyHeadersMiddleware() {\n    return async (req, res, next) => {\n      const maxUploadSizeOverride = req.get('X-Parse-File-Max-Upload-Size');\n      if (!maxUploadSizeOverride) {\n        return next();\n      }\n      const appId = req.get('X-Parse-Application-Id');\n      const config = Config.get(appId);\n      if (!config) {\n        const error = createSanitizedHttpError(403, 'Invalid application ID.', undefined);\n        res.status(error.status);\n        res.json({ error: error.message });\n        return;\n      }\n      const masterKey = await config.loadMasterKey();\n      if (req.get('X-Parse-Master-Key') !== masterKey) {\n        const error = createSanitizedHttpError(403, 'unauthorized: master key is required', config);\n        res.status(error.status);\n        res.json({ error: error.message });\n        return;\n      }\n      if (config.masterKeyIps?.length && !Middlewares.checkIp(req.ip, config.masterKeyIps, config.masterKeyIpsStore)) {\n        const error = createSanitizedHttpError(403, 'unauthorized: master key is required', config);\n        res.status(error.status);\n        res.json({ error: error.message });\n        return;\n      }\n      let parsedBytes;\n      try {\n        parsedBytes = Utils.parseSizeToBytes(maxUploadSizeOverride);\n      } catch {\n        return next(\n          new Parse.Error(\n            Parse.Error.FILE_SAVE_ERROR,\n            `Invalid maxUploadSize override value: ${maxUploadSizeOverride}`\n          )\n        );\n      }\n      req._maxUploadSizeOverride = parsedBytes;\n      next();\n    };\n  }\n\n  _bodyParsingMiddleware(maxUploadSize) {\n    const defaultMaxBytes = Utils.parseSizeToBytes(maxUploadSize);\n    return (req, res, next) => {\n      if (req.get('X-Parse-Upload-Mode') === 'stream') {\n        req._maxUploadSizeBytes = req._maxUploadSizeOverride ?? defaultMaxBytes;\n        return next();\n      }\n      const limit = req._maxUploadSizeOverride ?? maxUploadSize;\n      return express.raw({ type: () => true, limit })(req, res, next);\n    };\n  }\n\n  async createHandler(req, res, next) {\n    if (req.auth.isReadOnly) {\n      const error = createSanitizedHttpError(403, \"read-only masterKey isn't allowed to create a file.\", req.config);\n      res.status(error.status);\n      res.end(`{\"error\":\"${error.message}\"}`);\n      return;\n    }\n    const config = req.config;\n    const user = req.auth.user;\n    const isMaster = req.auth.isMaster;\n    const isLinked = user && Parse.AnonymousUtils.isLinked(user);\n    if (!isMaster && !config.fileUpload.enableForAnonymousUser && isLinked) {\n      next(\n        new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'File upload by anonymous user is disabled.')\n      );\n      return;\n    }\n    if (!isMaster && !config.fileUpload.enableForAuthenticatedUser && !isLinked && user) {\n      next(\n        new Parse.Error(\n          Parse.Error.FILE_SAVE_ERROR,\n          'File upload by authenticated user is disabled.'\n        )\n      );\n      return;\n    }\n    if (!isMaster && !config.fileUpload.enableForPublic && !user) {\n      next(new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'File upload by public is disabled.'));\n      return;\n    }\n    const filesController = config.filesController;\n    const { filename } = req.params;\n    const contentType = req.get('Content-type');\n\n    const error = filesController.validateFilename(filename);\n    if (error) {\n      next(error);\n      return;\n    }\n\n    const fileExtensions = config.fileUpload?.fileExtensions;\n    if (!isMaster && fileExtensions) {\n      const isValidExtension = extension => {\n        return fileExtensions.some(ext => {\n          if (ext === '*') {\n            return true;\n          }\n          const regex = new RegExp(ext);\n          if (regex.test(extension)) {\n            return true;\n          }\n        });\n      };\n      let extension = contentType;\n      if (filename && filename.includes('.')) {\n        extension = filename.substring(filename.lastIndexOf('.') + 1);\n      } else if (contentType && contentType.includes('/')) {\n        extension = contentType.split('/')[1];\n      }\n      // Strip MIME parameters (e.g. \";charset=utf-8\") and whitespace\n      extension = extension?.split(';')[0]?.replace(/\\s+/g, '');\n\n      if (extension && !isValidExtension(extension)) {\n        next(\n          new Parse.Error(\n            Parse.Error.FILE_SAVE_ERROR,\n            `File upload of extension ${extension} is disabled.`\n          )\n        );\n        return;\n      }\n    }\n\n    // For streaming uploads, read file data from headers since the body is the raw stream\n    if (req.get('X-Parse-Upload-Mode') === 'stream') {\n      req.fileData = {};\n      if (req.get('X-Parse-File-Directory')) {\n        req.fileData.directory = req.get('X-Parse-File-Directory');\n      }\n      if (req.get('X-Parse-File-Metadata')) {\n        try {\n          const parsed = JSON.parse(req.get('X-Parse-File-Metadata'));\n          if (!parsed || typeof parsed !== 'object' || Array.isArray(parsed)) {\n            throw new Error();\n          }\n          req.fileData.metadata = parsed;\n        } catch {\n          next(new Parse.Error(Parse.Error.INVALID_JSON, 'Invalid JSON in X-Parse-File-Metadata header.'));\n          return;\n        }\n      }\n      if (req.get('X-Parse-File-Tags')) {\n        try {\n          const parsed = JSON.parse(req.get('X-Parse-File-Tags'));\n          if (!parsed || typeof parsed !== 'object' || Array.isArray(parsed)) {\n            throw new Error();\n          }\n          req.fileData.tags = parsed;\n        } catch {\n          next(new Parse.Error(Parse.Error.INVALID_JSON, 'Invalid JSON in X-Parse-File-Tags header.'));\n          return;\n        }\n      }\n    }\n\n    // Validate directory option (requires master key)\n    const directory = req.fileData?.directory;\n    if (directory !== undefined) {\n      if (!isMaster) {\n        next(\n          new Parse.Error(\n            Parse.Error.OPERATION_FORBIDDEN,\n            'Directory can only be set using the Master Key.'\n          )\n        );\n        return;\n      }\n      const directoryError = FilesRouter.validateDirectory(directory);\n      if (directoryError) {\n        next(directoryError);\n        return;\n      }\n    }\n\n    // Dispatch to the appropriate handler based on whether the body was buffered\n    if (Buffer.isBuffer(req.body)) {\n      return this._handleBufferedUpload(req, res, next);\n    }\n    return this._handleStreamUpload(req, res, next);\n  }\n\n  async _handleBufferedUpload(req, res, next) {\n    const config = req.config;\n    const filesController = config.filesController;\n    const { filename } = req.params;\n    const contentType = req.get('Content-type');\n\n    if (!req.body || !req.body.length) {\n      next(new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'Invalid file upload.'));\n      return;\n    }\n\n    const base64 = req.body.toString('base64');\n    const file = new Parse.File(filename, { base64 }, contentType);\n    const { metadata = {}, tags = {}, directory } = req.fileData || {};\n    try {\n      // Scan request data for denied keywords\n      Utils.checkProhibitedKeywords(config, metadata);\n      Utils.checkProhibitedKeywords(config, tags);\n    } catch (error) {\n      next(new Parse.Error(Parse.Error.INVALID_KEY_NAME, error));\n      return;\n    }\n    file.setTags(tags);\n    file.setMetadata(metadata);\n    if (directory) {\n      file.setDirectory(directory);\n    }\n    const fileSize = Buffer.byteLength(req.body);\n    const fileObject = { file, fileSize };\n    try {\n      // run beforeSaveFile trigger\n      const triggerResult = await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeSave,\n        fileObject,\n        config,\n        req.auth\n      );\n      let saveResult;\n      // if a new ParseFile is returned check if it's an already saved file\n      if (triggerResult instanceof Parse.File) {\n        fileObject.file = triggerResult;\n        if (triggerResult.url()) {\n          // set fileSize to null because we wont know how big it is here\n          fileObject.fileSize = null;\n          saveResult = {\n            url: triggerResult.url(),\n            name: triggerResult._name,\n          };\n        }\n      }\n      // if the file returned by the trigger has already been saved skip saving anything\n      if (!saveResult) {\n        // update fileSize\n        let bufferData;\n        if (fileObject.file._source?.format === 'buffer') {\n          bufferData = fileObject.file._source.buffer;\n        } else {\n          bufferData = Buffer.from(fileObject.file._data, 'base64');\n        }\n        fileObject.fileSize = Buffer.byteLength(bufferData);\n        // prepare file options\n        const fileOptions = {\n          metadata: fileObject.file._metadata,\n        };\n        // some s3-compatible providers (DigitalOcean, Linode) do not accept tags\n        // so we do not include the tags option if it is empty.\n        const fileTags =\n          Object.keys(fileObject.file._tags).length > 0 ? { tags: fileObject.file._tags } : {};\n        Object.assign(fileOptions, fileTags);\n        // include directory if set (from client request or beforeSaveFile trigger)\n        if (fileObject.file._directory) {\n          fileOptions.directory = fileObject.file._directory;\n        }\n        // save file\n        const createFileResult = await filesController.createFile(\n          config,\n          fileObject.file._name,\n          bufferData,\n          fileObject.file._source.type,\n          fileOptions\n        );\n        // update file with new data\n        fileObject.file._name = createFileResult.name;\n        fileObject.file._url = createFileResult.url;\n        fileObject.file._requestTask = null;\n        fileObject.file._previousSave = Promise.resolve(fileObject.file);\n        saveResult = {\n          url: createFileResult.url,\n          name: createFileResult.name,\n        };\n      }\n      // run afterSaveFile trigger\n      await triggers.maybeRunFileTrigger(triggers.Types.afterSave, fileObject, config, req.auth);\n      res.status(201);\n      res.set('Location', saveResult.url);\n      res.json(saveResult);\n    } catch (e) {\n      logger.error('Error creating a file: ', e);\n      const error = triggers.resolveError(e, {\n        code: Parse.Error.FILE_SAVE_ERROR,\n        message: `Could not store file: ${fileObject.file._name}.`,\n      });\n      next(error);\n    }\n  }\n\n  async _handleStreamUpload(req, res, next) {\n    const config = req.config;\n    const filesController = config.filesController;\n    const { filename } = req.params;\n    let contentType = req.get('Content-Type');\n    const maxBytes = req._maxUploadSizeBytes;\n    let stream;\n\n    try {\n      // Early rejection via Content-Length header\n      const contentLength = req.get('Content-Length');\n      if (contentLength && parseInt(contentLength, 10) > maxBytes) {\n        req.resume();\n        next(new Parse.Error(\n          Parse.Error.FILE_SAVE_ERROR,\n          `File size exceeds maximum allowed: ${maxBytes} bytes.`\n        ));\n        return;\n      }\n\n      const mime = (await import('mime')).default;\n\n      // Infer content type from extension or add extension from content type\n      const hasExtension = filename && filename.includes('.');\n      if (hasExtension && !contentType) {\n        contentType = mime.getType(filename);\n      } else if (!hasExtension && contentType) {\n        // extension will be added by filesController.createFile\n      }\n\n      // Create size-limited stream wrapping the request\n      stream = createSizeLimitedStream(req, maxBytes);\n\n      // Build a Parse.File with no _data (streaming mode)\n      const file = new Parse.File(filename, { base64: '' }, contentType);\n      const { metadata = {}, tags = {}, directory } = req.fileData || {};\n\n      // Validate metadata and tags for prohibited keywords\n      try {\n        Utils.checkProhibitedKeywords(config, metadata);\n        Utils.checkProhibitedKeywords(config, tags);\n      } catch (error) {\n        stream.destroy();\n        next(new Parse.Error(Parse.Error.INVALID_KEY_NAME, error));\n        return;\n      }\n\n      file.setTags(tags);\n      file.setMetadata(metadata);\n      if (directory) {\n        file.setDirectory(directory);\n      }\n\n      const fileSize = req.get('Content-Length')\n        ? parseInt(req.get('Content-Length'), 10)\n        : null;\n      const fileObject = { file, fileSize, stream: true };\n\n      // Run beforeSaveFile trigger\n      const triggerResult = await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeSave,\n        fileObject,\n        config,\n        req.auth\n      );\n\n      let saveResult;\n      // If a new ParseFile is returned, check if it's an already saved file\n      if (triggerResult instanceof Parse.File) {\n        fileObject.file = triggerResult;\n        if (triggerResult.url()) {\n          fileObject.fileSize = null;\n          saveResult = {\n            url: triggerResult.url(),\n            name: triggerResult._name,\n          };\n          // Destroy stream to remove listeners and drain request\n          stream.destroy();\n        }\n      }\n\n      // If the file returned by the trigger has already been saved, skip saving\n      if (!saveResult) {\n        // Prepare file options\n        const fileOptions = {\n          metadata: fileObject.file._metadata,\n        };\n        const fileTags =\n          Object.keys(fileObject.file._tags).length > 0 ? { tags: fileObject.file._tags } : {};\n        Object.assign(fileOptions, fileTags);\n        // include directory if set (from client request or beforeSaveFile trigger)\n        if (fileObject.file._directory) {\n          fileOptions.directory = fileObject.file._directory;\n        }\n\n        // Pass stream directly to filesController — it will buffer if adapter doesn't support streaming\n        const sourceType = fileObject.file._source?.type || contentType;\n        const createFileResult = await filesController.createFile(\n          config,\n          fileObject.file._name,\n          stream,\n          sourceType,\n          fileOptions\n        );\n\n        // Update file with new data\n        fileObject.file._name = createFileResult.name;\n        fileObject.file._url = createFileResult.url;\n        fileObject.file._requestTask = null;\n        fileObject.file._previousSave = Promise.resolve(fileObject.file);\n        saveResult = {\n          url: createFileResult.url,\n          name: createFileResult.name,\n        };\n      }\n\n      // Run afterSaveFile trigger\n      await triggers.maybeRunFileTrigger(triggers.Types.afterSave, fileObject, config, req.auth);\n      res.status(201);\n      res.set('Location', saveResult.url);\n      res.json(saveResult);\n    } catch (e) {\n      // Destroy stream to remove listeners and drain request, or resume directly\n      if (stream) {\n        stream.destroy();\n      } else {\n        req.resume();\n      }\n      logger.error('Error creating a file: ', e);\n      const error = triggers.resolveError(e, {\n        code: Parse.Error.FILE_SAVE_ERROR,\n        message: `Could not store file: ${filename}.`,\n      });\n      next(error);\n    }\n  }\n\n  async deleteHandler(req, res, next) {\n    if (req.auth.isReadOnly) {\n      const error = createSanitizedHttpError(403, \"read-only masterKey isn't allowed to delete a file.\", req.config);\n      res.status(error.status);\n      res.end(`{\"error\":\"${error.message}\"}`);\n      return;\n    }\n    try {\n      const { filesController } = req.config;\n      const filename = FilesRouter._getFilenameFromParams(req);\n      // run beforeDeleteFile trigger\n      const file = new Parse.File(filename);\n      file._url = await filesController.adapter.getFileLocation(req.config, filename);\n      const fileObject = { file, fileSize: null };\n      await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeDelete,\n        fileObject,\n        req.config,\n        req.auth\n      );\n      // delete file\n      await filesController.deleteFile(req.config, filename);\n      // run afterDeleteFile trigger\n      await triggers.maybeRunFileTrigger(\n        triggers.Types.afterDelete,\n        fileObject,\n        req.config,\n        req.auth\n      );\n      res.status(200);\n      // TODO: return useful JSON here?\n      res.end();\n    } catch (e) {\n      logger.error('Error deleting a file: ', e);\n      const error = triggers.resolveError(e, {\n        code: Parse.Error.FILE_DELETE_ERROR,\n        message: 'Could not delete file.',\n      });\n      next(error);\n    }\n  }\n\n  async metadataHandler(req, res) {\n    try {\n      const config = Config.get(req.params.appId);\n      if (!config) {\n        res.status(200);\n        res.json({});\n        return;\n      }\n      const { filesController } = config;\n      let filename = FilesRouter._getFilenameFromParams(req);\n      const file = new Parse.File(filename, { base64: '' });\n      const fileAuth = await FilesRouter._resolveAuth(req, config);\n      const triggerResult = await triggers.maybeRunFileTrigger(\n        triggers.Types.beforeFind,\n        { file },\n        config,\n        fileAuth\n      );\n      if (triggerResult?.file?._name) {\n        filename = triggerResult.file._name;\n      }\n      const data = await filesController.getMetadata(filename).catch(() => {\n        res.status(200);\n        res.json({});\n      });\n      if (!data) {\n        return;\n      }\n      await triggers.maybeRunFileTrigger(\n        triggers.Types.afterFind,\n        { file },\n        config,\n        fileAuth\n      );\n      res.status(200);\n      res.json(data);\n    } catch (e) {\n      const err = triggers.resolveError(e, {\n        code: Parse.Error.SCRIPT_FAILED,\n        message: 'Could not get file metadata.',\n      });\n      res.status(403);\n      res.json({ code: err.code, error: err.message });\n    }\n  }\n}\n\nfunction isFileStreamable(req, filesController) {\n  const range = (req.get('Range') || '/-/').split('-');\n  const start = Number(range[0]);\n  const end = Number(range[1]);\n  return (\n    (!isNaN(start) || !isNaN(end)) && typeof filesController.adapter.handleFileStream === 'function'\n  );\n}\n"],"mappings":"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","ignoreList":[]}
|
package/lib/batch.js
CHANGED
|
@@ -83,6 +83,9 @@ async function handleBatch(router, req) {
|
|
|
83
83
|
const rateLimits = req.config.rateLimits || [];
|
|
84
84
|
for (const restRequest of req.body.requests) {
|
|
85
85
|
const routablePath = makeRoutablePath(restRequest.path);
|
|
86
|
+
if ((restRequest.method || 'GET').toUpperCase() === 'POST' && routablePath === batchPath) {
|
|
87
|
+
throw new Parse.Error(Parse.Error.INVALID_JSON, 'nested batch requests are not allowed');
|
|
88
|
+
}
|
|
86
89
|
for (const limit of rateLimits) {
|
|
87
90
|
const pathExp = limit.path.regexp || limit.path;
|
|
88
91
|
if (!pathExp.test(routablePath)) {
|
|
@@ -179,4 +182,4 @@ module.exports = {
|
|
|
179
182
|
mountOnto,
|
|
180
183
|
makeBatchRoutingPathFunction
|
|
181
184
|
};
|
|
182
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","path","batchPath","mountOnto","router","route","req","handleBatch","parseURL","urlString","URL","undefined","makeBatchRoutingPathFunction","originalUrl","serverURL","publicServerURL","apiPrefixLength","length","apiPrefix","slice","makeRoutablePath","requestPath","Error","INVALID_JSON","posix","join","pathname","localPath","publicPath","startsWithLocal","startsWith","startsWithPublic","pathLengthToUse","Math","max","newPath","Array","isArray","body","requests","batchRequestLimit","config","requestComplexity","auth","isMaster","isMaintenance","restRequest","endsWith","rateLimits","routablePath","limit","pathExp","regexp","test","info","sessionToken","fakeReq","ip","method","toUpperCase","_batchOriginalMethod","fakeRes","setHeader","handler","err","CONNECTION_FAILED","errorResponseMessage","batch","transactionRetries","initialPromise","Promise","resolve","transaction","database","createTransactionalSession","then","promises","map","request","tryRouteRequest","response","success","error","code","message","all","results","find","result","abortTransactionalSession","reject","commitTransactionalSession","catch","errorItem","module","exports"],"sources":["../src/batch.js"],"sourcesContent":["const Parse = require('parse/node').Parse;\nconst path = require('path');\n// These methods handle batch requests.\nconst batchPath = '/batch';\n\n// Mounts a batch-handler onto a PromiseRouter.\nfunction mountOnto(router) {\n  router.route('POST', batchPath, req => {\n    return handleBatch(router, req);\n  });\n}\n\nfunction parseURL(urlString) {\n  try {\n    return new URL(urlString);\n  } catch {\n    return undefined;\n  }\n}\n\nfunction makeBatchRoutingPathFunction(originalUrl, serverURL, publicServerURL) {\n  serverURL = serverURL ? parseURL(serverURL) : undefined;\n  publicServerURL = publicServerURL ? parseURL(publicServerURL) : undefined;\n\n  const apiPrefixLength = originalUrl.length - batchPath.length;\n  let apiPrefix = originalUrl.slice(0, apiPrefixLength);\n\n  const makeRoutablePath = function (requestPath) {\n    // The routablePath is the path minus the api prefix\n    if (requestPath.slice(0, apiPrefix.length) != apiPrefix) {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, 'cannot route batch path ' + requestPath);\n    }\n    return path.posix.join('/', requestPath.slice(apiPrefix.length));\n  };\n\n  if (serverURL && publicServerURL && serverURL.pathname != publicServerURL.pathname) {\n    const localPath = serverURL.pathname;\n    const publicPath = publicServerURL.pathname;\n\n    // Override the api prefix\n    apiPrefix = localPath;\n    return function (requestPath) {\n      // Figure out which server url was used by figuring out which\n      // path more closely matches requestPath\n      const startsWithLocal = requestPath.startsWith(localPath);\n      const startsWithPublic = requestPath.startsWith(publicPath);\n      const pathLengthToUse =\n        startsWithLocal && startsWithPublic\n          ? Math.max(localPath.length, publicPath.length)\n          : startsWithLocal\n            ? localPath.length\n            : publicPath.length;\n\n      const newPath = path.posix.join('/', localPath, '/', requestPath.slice(pathLengthToUse));\n\n      // Use the method for local routing\n      return makeRoutablePath(newPath);\n    };\n  }\n\n  return makeRoutablePath;\n}\n\n// Returns a promise for a {response} object.\n// TODO: pass along auth correctly\nasync function handleBatch(router, req) {\n  if (!Array.isArray(req.body?.requests)) {\n    throw new Parse.Error(Parse.Error.INVALID_JSON, 'requests must be an array');\n  }\n  const batchRequestLimit = req.config?.requestComplexity?.batchRequestLimit ?? -1;\n  if (batchRequestLimit > -1 && !req.auth?.isMaster && !req.auth?.isMaintenance && req.body.requests.length > batchRequestLimit) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `Batch request contains ${req.body.requests.length} sub-requests, which exceeds the limit of ${batchRequestLimit}.`\n    );\n  }\n  for (const restRequest of req.body.requests) {\n    if (!restRequest || typeof restRequest !== 'object' || typeof restRequest.path !== 'string') {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, 'batch request path must be a string');\n    }\n  }\n\n  // The batch paths are all from the root of our domain.\n  // That means they include the API prefix, that the API is mounted\n  // to. However, our promise router does not route the api prefix. So\n  // we need to figure out the API prefix, so that we can strip it\n  // from all the subrequests.\n  if (!req.originalUrl.endsWith(batchPath)) {\n    throw 'internal routing problem - expected url to end with batch';\n  }\n\n  const makeRoutablePath = makeBatchRoutingPathFunction(\n    req.originalUrl,\n    req.config.serverURL,\n    req.config.publicServerURL\n  );\n\n  // Enforce rate limits for each batch sub-request by invoking the\n  // rate limit handler. This ensures sub-requests consume tokens from\n  // the same window state as direct requests.\n  const rateLimits = req.config.rateLimits || [];\n  for (const restRequest of req.body.requests) {\n    const routablePath = makeRoutablePath(restRequest.path);\n    for (const limit of rateLimits) {\n      const pathExp = limit.path.regexp || limit.path;\n      if (!pathExp.test(routablePath)) {\n        continue;\n      }\n      const info = { ...req.info };\n      if (routablePath === '/login') {\n        delete info.sessionToken;\n      }\n      const fakeReq = {\n        ip: req.ip || req.config?.ip || '127.0.0.1',\n        method: (restRequest.method || 'GET').toUpperCase(),\n        _batchOriginalMethod: 'POST',\n        config: req.config,\n        auth: req.auth,\n        info,\n      };\n      const fakeRes = { setHeader() {} };\n      try {\n        await limit.handler(fakeReq, fakeRes, err => {\n          if (err) {\n            throw err;\n          }\n        });\n      } catch {\n        throw new Parse.Error(\n          Parse.Error.CONNECTION_FAILED,\n          limit.errorResponseMessage || 'Too many requests'\n        );\n      }\n    }\n  }\n\n  const batch = transactionRetries => {\n    let initialPromise = Promise.resolve();\n    if (req.body?.transaction === true) {\n      initialPromise = req.config.database.createTransactionalSession();\n    }\n\n    return initialPromise.then(() => {\n      const promises = req.body?.requests.map(restRequest => {\n        const routablePath = makeRoutablePath(restRequest.path);\n\n        // Construct a request that we can send to a handler\n        const request = {\n          body: restRequest.body,\n          config: req.config,\n          auth: req.auth,\n          info: req.info,\n        };\n\n        return router.tryRouteRequest(restRequest.method, routablePath, request).then(\n          response => {\n            return { success: response.response };\n          },\n          error => {\n            return { error: { code: error.code, error: error.message } };\n          }\n        );\n      });\n\n      return Promise.all(promises)\n        .then(results => {\n          if (req.body?.transaction === true) {\n            if (results.find(result => typeof result.error === 'object')) {\n              return req.config.database.abortTransactionalSession().then(() => {\n                return Promise.reject({ response: results });\n              });\n            } else {\n              return req.config.database.commitTransactionalSession().then(() => {\n                return { response: results };\n              });\n            }\n          } else {\n            return { response: results };\n          }\n        })\n        .catch(error => {\n          if (\n            error &&\n            error.response &&\n            error.response.find(\n              errorItem => typeof errorItem.error === 'object' && errorItem.error.code === 251\n            ) &&\n            transactionRetries > 0\n          ) {\n            return batch(transactionRetries - 1);\n          }\n          throw error;\n        });\n    });\n  };\n  return batch(5);\n}\n\nmodule.exports = {\n  mountOnto,\n  makeBatchRoutingPathFunction,\n};\n"],"mappings":"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","ignoreList":[]}
|
|
185
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["Parse","require","path","batchPath","mountOnto","router","route","req","handleBatch","parseURL","urlString","URL","undefined","makeBatchRoutingPathFunction","originalUrl","serverURL","publicServerURL","apiPrefixLength","length","apiPrefix","slice","makeRoutablePath","requestPath","Error","INVALID_JSON","posix","join","pathname","localPath","publicPath","startsWithLocal","startsWith","startsWithPublic","pathLengthToUse","Math","max","newPath","Array","isArray","body","requests","batchRequestLimit","config","requestComplexity","auth","isMaster","isMaintenance","restRequest","endsWith","rateLimits","routablePath","method","toUpperCase","limit","pathExp","regexp","test","info","sessionToken","fakeReq","ip","_batchOriginalMethod","fakeRes","setHeader","handler","err","CONNECTION_FAILED","errorResponseMessage","batch","transactionRetries","initialPromise","Promise","resolve","transaction","database","createTransactionalSession","then","promises","map","request","tryRouteRequest","response","success","error","code","message","all","results","find","result","abortTransactionalSession","reject","commitTransactionalSession","catch","errorItem","module","exports"],"sources":["../src/batch.js"],"sourcesContent":["const Parse = require('parse/node').Parse;\nconst path = require('path');\n// These methods handle batch requests.\nconst batchPath = '/batch';\n\n// Mounts a batch-handler onto a PromiseRouter.\nfunction mountOnto(router) {\n  router.route('POST', batchPath, req => {\n    return handleBatch(router, req);\n  });\n}\n\nfunction parseURL(urlString) {\n  try {\n    return new URL(urlString);\n  } catch {\n    return undefined;\n  }\n}\n\nfunction makeBatchRoutingPathFunction(originalUrl, serverURL, publicServerURL) {\n  serverURL = serverURL ? parseURL(serverURL) : undefined;\n  publicServerURL = publicServerURL ? parseURL(publicServerURL) : undefined;\n\n  const apiPrefixLength = originalUrl.length - batchPath.length;\n  let apiPrefix = originalUrl.slice(0, apiPrefixLength);\n\n  const makeRoutablePath = function (requestPath) {\n    // The routablePath is the path minus the api prefix\n    if (requestPath.slice(0, apiPrefix.length) != apiPrefix) {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, 'cannot route batch path ' + requestPath);\n    }\n    return path.posix.join('/', requestPath.slice(apiPrefix.length));\n  };\n\n  if (serverURL && publicServerURL && serverURL.pathname != publicServerURL.pathname) {\n    const localPath = serverURL.pathname;\n    const publicPath = publicServerURL.pathname;\n\n    // Override the api prefix\n    apiPrefix = localPath;\n    return function (requestPath) {\n      // Figure out which server url was used by figuring out which\n      // path more closely matches requestPath\n      const startsWithLocal = requestPath.startsWith(localPath);\n      const startsWithPublic = requestPath.startsWith(publicPath);\n      const pathLengthToUse =\n        startsWithLocal && startsWithPublic\n          ? Math.max(localPath.length, publicPath.length)\n          : startsWithLocal\n            ? localPath.length\n            : publicPath.length;\n\n      const newPath = path.posix.join('/', localPath, '/', requestPath.slice(pathLengthToUse));\n\n      // Use the method for local routing\n      return makeRoutablePath(newPath);\n    };\n  }\n\n  return makeRoutablePath;\n}\n\n// Returns a promise for a {response} object.\n// TODO: pass along auth correctly\nasync function handleBatch(router, req) {\n  if (!Array.isArray(req.body?.requests)) {\n    throw new Parse.Error(Parse.Error.INVALID_JSON, 'requests must be an array');\n  }\n  const batchRequestLimit = req.config?.requestComplexity?.batchRequestLimit ?? -1;\n  if (batchRequestLimit > -1 && !req.auth?.isMaster && !req.auth?.isMaintenance && req.body.requests.length > batchRequestLimit) {\n    throw new Parse.Error(\n      Parse.Error.INVALID_JSON,\n      `Batch request contains ${req.body.requests.length} sub-requests, which exceeds the limit of ${batchRequestLimit}.`\n    );\n  }\n  for (const restRequest of req.body.requests) {\n    if (!restRequest || typeof restRequest !== 'object' || typeof restRequest.path !== 'string') {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, 'batch request path must be a string');\n    }\n  }\n\n  // The batch paths are all from the root of our domain.\n  // That means they include the API prefix, that the API is mounted\n  // to. However, our promise router does not route the api prefix. So\n  // we need to figure out the API prefix, so that we can strip it\n  // from all the subrequests.\n  if (!req.originalUrl.endsWith(batchPath)) {\n    throw 'internal routing problem - expected url to end with batch';\n  }\n\n  const makeRoutablePath = makeBatchRoutingPathFunction(\n    req.originalUrl,\n    req.config.serverURL,\n    req.config.publicServerURL\n  );\n\n  // Enforce rate limits for each batch sub-request by invoking the\n  // rate limit handler. This ensures sub-requests consume tokens from\n  // the same window state as direct requests.\n  const rateLimits = req.config.rateLimits || [];\n  for (const restRequest of req.body.requests) {\n    const routablePath = makeRoutablePath(restRequest.path);\n    if ((restRequest.method || 'GET').toUpperCase() === 'POST' && routablePath === batchPath) {\n      throw new Parse.Error(Parse.Error.INVALID_JSON, 'nested batch requests are not allowed');\n    }\n    for (const limit of rateLimits) {\n      const pathExp = limit.path.regexp || limit.path;\n      if (!pathExp.test(routablePath)) {\n        continue;\n      }\n      const info = { ...req.info };\n      if (routablePath === '/login') {\n        delete info.sessionToken;\n      }\n      const fakeReq = {\n        ip: req.ip || req.config?.ip || '127.0.0.1',\n        method: (restRequest.method || 'GET').toUpperCase(),\n        _batchOriginalMethod: 'POST',\n        config: req.config,\n        auth: req.auth,\n        info,\n      };\n      const fakeRes = { setHeader() {} };\n      try {\n        await limit.handler(fakeReq, fakeRes, err => {\n          if (err) {\n            throw err;\n          }\n        });\n      } catch {\n        throw new Parse.Error(\n          Parse.Error.CONNECTION_FAILED,\n          limit.errorResponseMessage || 'Too many requests'\n        );\n      }\n    }\n  }\n\n  const batch = transactionRetries => {\n    let initialPromise = Promise.resolve();\n    if (req.body?.transaction === true) {\n      initialPromise = req.config.database.createTransactionalSession();\n    }\n\n    return initialPromise.then(() => {\n      const promises = req.body?.requests.map(restRequest => {\n        const routablePath = makeRoutablePath(restRequest.path);\n\n        // Construct a request that we can send to a handler\n        const request = {\n          body: restRequest.body,\n          config: req.config,\n          auth: req.auth,\n          info: req.info,\n        };\n\n        return router.tryRouteRequest(restRequest.method, routablePath, request).then(\n          response => {\n            return { success: response.response };\n          },\n          error => {\n            return { error: { code: error.code, error: error.message } };\n          }\n        );\n      });\n\n      return Promise.all(promises)\n        .then(results => {\n          if (req.body?.transaction === true) {\n            if (results.find(result => typeof result.error === 'object')) {\n              return req.config.database.abortTransactionalSession().then(() => {\n                return Promise.reject({ response: results });\n              });\n            } else {\n              return req.config.database.commitTransactionalSession().then(() => {\n                return { response: results };\n              });\n            }\n          } else {\n            return { response: results };\n          }\n        })\n        .catch(error => {\n          if (\n            error &&\n            error.response &&\n            error.response.find(\n              errorItem => typeof errorItem.error === 'object' && errorItem.error.code === 251\n            ) &&\n            transactionRetries > 0\n          ) {\n            return batch(transactionRetries - 1);\n          }\n          throw error;\n        });\n    });\n  };\n  return batch(5);\n}\n\nmodule.exports = {\n  mountOnto,\n  makeBatchRoutingPathFunction,\n};\n"],"mappings":"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","ignoreList":[]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "parse-server",
|
|
3
|
-
"version": "9.7.
|
|
3
|
+
"version": "9.7.1-alpha.2",
|
|
4
4
|
"description": "An express module providing a Parse-compatible API server",
|
|
5
5
|
"main": "lib/index.js",
|
|
6
6
|
"repository": {
|
|
@@ -31,7 +31,7 @@
|
|
|
31
31
|
"commander": "14.0.3",
|
|
32
32
|
"cors": "2.8.6",
|
|
33
33
|
"express": "5.2.1",
|
|
34
|
-
"express-rate-limit": "8.3.
|
|
34
|
+
"express-rate-limit": "8.3.1",
|
|
35
35
|
"follow-redirects": "1.15.11",
|
|
36
36
|
"graphql": "16.13.2",
|
|
37
37
|
"graphql-list-fields": "2.0.4",
|
|
@@ -43,7 +43,7 @@
|
|
|
43
43
|
"ldapjs": "3.0.7",
|
|
44
44
|
"lodash": "4.17.23",
|
|
45
45
|
"lru-cache": "11.2.7",
|
|
46
|
-
"mime": "4.0
|
|
46
|
+
"mime": "4.1.0",
|
|
47
47
|
"mongodb": "7.1.0",
|
|
48
48
|
"mustache": "4.2.0",
|
|
49
49
|
"otpauth": "9.4.0",
|
|
@@ -57,7 +57,7 @@
|
|
|
57
57
|
"redis": "5.11.0",
|
|
58
58
|
"semver": "7.7.2",
|
|
59
59
|
"tv4": "1.3.0",
|
|
60
|
-
"uuid": "
|
|
60
|
+
"uuid": "13.0.0",
|
|
61
61
|
"winston": "3.19.0",
|
|
62
62
|
"winston-daily-rotate-file": "5.0.0",
|
|
63
63
|
"ws": "8.20.0"
|
|
@@ -70,7 +70,7 @@
|
|
|
70
70
|
"@babel/eslint-parser": "7.28.6",
|
|
71
71
|
"@babel/plugin-proposal-object-rest-spread": "7.20.7",
|
|
72
72
|
"@babel/plugin-transform-flow-strip-types": "7.27.1",
|
|
73
|
-
"@babel/preset-env": "7.
|
|
73
|
+
"@babel/preset-env": "7.29.2",
|
|
74
74
|
"@babel/preset-typescript": "7.27.1",
|
|
75
75
|
"@saithodev/semantic-release-backmerge": "4.0.1",
|
|
76
76
|
"@semantic-release/changelog": "6.0.3",
|
|
@@ -94,7 +94,7 @@
|
|
|
94
94
|
"jasmine-spec-reporter": "7.0.0",
|
|
95
95
|
"jsdoc": "4.0.4",
|
|
96
96
|
"jsdoc-babel": "0.5.0",
|
|
97
|
-
"lint-staged": "16.
|
|
97
|
+
"lint-staged": "16.4.0",
|
|
98
98
|
"m": "1.10.0",
|
|
99
99
|
"madge": "8.0.0",
|
|
100
100
|
"mock-files-adapter": "file:spec/dependencies/mock-files-adapter",
|
|
@@ -107,7 +107,7 @@
|
|
|
107
107
|
"semantic-release": "25.0.3",
|
|
108
108
|
"typescript": "5.9.3",
|
|
109
109
|
"typescript-eslint": "8.53.1",
|
|
110
|
-
"yaml": "2.8.
|
|
110
|
+
"yaml": "2.8.3"
|
|
111
111
|
},
|
|
112
112
|
"scripts": {
|
|
113
113
|
"ci:check": "node ./ci/ciCheck.js",
|