parse-server 9.8.0-alpha.2 → 9.8.0-alpha.4

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.
@@ -16,7 +16,6 @@ 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');
20
19
  /**
21
20
  * Wraps a readable stream in a Readable that enforces a byte size limit.
22
21
  * Data flow is lazy: the source is not read until a consumer starts reading
@@ -87,9 +86,29 @@ class FilesRouter {
87
86
  maxUploadSize = '20Mb'
88
87
  } = {}) {
89
88
  var router = _express.default.Router();
89
+ // Lightweight info initializer so handleParseSession can resolve session tokens.
90
+ // Unlike POST/DELETE routes, GET file routes skip handleParseHeaders (which
91
+ // normally sets req.info) because those requests may not carry Parse headers.
92
+ const initInfo = (req, res, next) => {
93
+ if (!req.info) {
94
+ const sessionToken = req.get('X-Parse-Session-Token');
95
+ req.info = {
96
+ sessionToken,
97
+ installationId: req.get('X-Parse-Installation-Id')
98
+ };
99
+ // If no session token and no auth yet (public access), set a minimal
100
+ // auth object so handleParseSession skips session resolution.
101
+ if (!sessionToken && !req.auth) {
102
+ req.auth = {
103
+ isMaster: false
104
+ };
105
+ }
106
+ }
107
+ next();
108
+ };
90
109
  // Metadata route must come before the catch-all GET route
91
- router.get('/files/:appId/metadata/*filepath', this.metadataHandler);
92
- router.get('/files/:appId/*filepath', this.getHandler);
110
+ router.get('/files/:appId/metadata/*filepath', initInfo, Middlewares.handleParseSession, this.metadataHandler);
111
+ router.get('/files/:appId/*filepath', initInfo, Middlewares.handleParseSession, this.getHandler);
93
112
  router.post('/files', function (req, res, next) {
94
113
  next(new _node.default.Error(_node.default.Error.INVALID_FILE_NAME, 'Filename not provided.'));
95
114
  });
@@ -101,21 +120,6 @@ class FilesRouter {
101
120
  const parts = req.params.filepath;
102
121
  return Array.isArray(parts) ? parts.join('/') : parts;
103
122
  }
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
- }
119
123
  static validateDirectory(directory) {
120
124
  if (typeof directory !== 'string') {
121
125
  return new _node.default.Error(_node.default.Error.INVALID_FILE_NAME, 'Directory must be a string.');
@@ -145,6 +149,24 @@ class FilesRouter {
145
149
  }
146
150
  return null;
147
151
  }
152
+ static _validateFileDownload(req, config) {
153
+ const isMaster = req.auth?.isMaster;
154
+ const isMaintenance = req.auth?.isMaintenance;
155
+ if (isMaster || isMaintenance) {
156
+ return;
157
+ }
158
+ const user = req.auth?.user;
159
+ const isLinked = user && _node.default.AnonymousUtils.isLinked(user);
160
+ if (!config.fileDownload.enableForAnonymousUser && isLinked) {
161
+ throw new _node.default.Error(_node.default.Error.OPERATION_FORBIDDEN, 'File download by anonymous user is disabled.');
162
+ }
163
+ if (!config.fileDownload.enableForAuthenticatedUser && !isLinked && user) {
164
+ throw new _node.default.Error(_node.default.Error.OPERATION_FORBIDDEN, 'File download by authenticated user is disabled.');
165
+ }
166
+ if (!config.fileDownload.enableForPublic && !user) {
167
+ throw new _node.default.Error(_node.default.Error.OPERATION_FORBIDDEN, 'File download by public is disabled.');
168
+ }
169
+ }
148
170
  async getHandler(req, res) {
149
171
  const config = _Config.default.get(req.params.appId);
150
172
  if (!config) {
@@ -155,6 +177,7 @@ class FilesRouter {
155
177
  });
156
178
  return;
157
179
  }
180
+ FilesRouter._validateFileDownload(req, config);
158
181
  let filename = FilesRouter._getFilenameFromParams(req);
159
182
  try {
160
183
  const filesController = config.filesController;
@@ -163,7 +186,7 @@ class FilesRouter {
163
186
  let file = new _node.default.File(filename, {
164
187
  base64: ''
165
188
  }, contentType);
166
- const fileAuth = await FilesRouter._resolveAuth(req, config);
189
+ const fileAuth = req.auth;
167
190
  const triggerResult = await triggers.maybeRunFileTrigger(triggers.Types.beforeFind, {
168
191
  file
169
192
  }, config, fileAuth);
@@ -316,20 +339,23 @@ class FilesRouter {
316
339
  return;
317
340
  }
318
341
  const config = req.config;
319
- const user = req.auth.user;
320
342
  const isMaster = req.auth.isMaster;
321
- const isLinked = user && _node.default.AnonymousUtils.isLinked(user);
322
- if (!isMaster && !config.fileUpload.enableForAnonymousUser && isLinked) {
323
- next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, 'File upload by anonymous user is disabled.'));
324
- return;
325
- }
326
- if (!isMaster && !config.fileUpload.enableForAuthenticatedUser && !isLinked && user) {
327
- next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, 'File upload by authenticated user is disabled.'));
328
- return;
329
- }
330
- if (!isMaster && !config.fileUpload.enableForPublic && !user) {
331
- next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, 'File upload by public is disabled.'));
332
- return;
343
+ const isMaintenance = req.auth.isMaintenance;
344
+ if (!isMaster && !isMaintenance) {
345
+ const user = req.auth.user;
346
+ const isLinked = user && _node.default.AnonymousUtils.isLinked(user);
347
+ if (!config.fileUpload.enableForAnonymousUser && isLinked) {
348
+ next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, 'File upload by anonymous user is disabled.'));
349
+ return;
350
+ }
351
+ if (!config.fileUpload.enableForAuthenticatedUser && !isLinked && user) {
352
+ next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, 'File upload by authenticated user is disabled.'));
353
+ return;
354
+ }
355
+ if (!config.fileUpload.enableForPublic && !user) {
356
+ next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, 'File upload by public is disabled.'));
357
+ return;
358
+ }
333
359
  }
334
360
  const filesController = config.filesController;
335
361
  const {
@@ -696,6 +722,7 @@ class FilesRouter {
696
722
  res.json({});
697
723
  return;
698
724
  }
725
+ FilesRouter._validateFileDownload(req, config);
699
726
  const {
700
727
  filesController
701
728
  } = config;
@@ -703,7 +730,7 @@ class FilesRouter {
703
730
  const file = new _node.default.File(filename, {
704
731
  base64: ''
705
732
  });
706
- const fileAuth = await FilesRouter._resolveAuth(req, config);
733
+ const fileAuth = req.auth;
707
734
  const triggerResult = await triggers.maybeRunFileTrigger(triggers.Types.beforeFind, {
708
735
  file
709
736
  }, config, fileAuth);
@@ -742,4 +769,4 @@ function isFileStreamable(req, filesController) {
742
769
  const end = Number(range[1]);
743
770
  return (!isNaN(start) || !isNaN(end)) && typeof filesController.adapter.handleFileStream === 'function';
744
771
  }
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":[]}
772
+ //# 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","initInfo","req","res","next","info","sessionToken","installationId","auth","isMaster","handleParseSession","metadataHandler","getHandler","post","INVALID_FILE_NAME","_earlyHeadersMiddleware","_bodyParsingMiddleware","handleParseHeaders","createHandler","bind","delete","enforceMasterKeyAccess","deleteHandler","_getFilenameFromParams","parts","params","filepath","Array","isArray","join","validateDirectory","directory","includes","startsWith","endsWith","firstSegment","split","dirRegex","test","_validateFileDownload","config","isMaintenance","user","isLinked","AnonymousUtils","fileDownload","enableForAnonymousUser","OPERATION_FORBIDDEN","enableForAuthenticatedUser","enableForPublic","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","fileUpload","validateFilename","fileExtensions","isValidExtension","extension","some","ext","regex","RegExp","substring","lastIndexOf","replace","fileData","parsed","JSON","parse","metadata","INVALID_JSON","tags","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    // Lightweight info initializer so handleParseSession can resolve session tokens.\n    // Unlike POST/DELETE routes, GET file routes skip handleParseHeaders (which\n    // normally sets req.info) because those requests may not carry Parse headers.\n    const initInfo = (req, res, next) => {\n      if (!req.info) {\n        const sessionToken = req.get('X-Parse-Session-Token');\n        req.info = {\n          sessionToken,\n          installationId: req.get('X-Parse-Installation-Id'),\n        };\n        // If no session token and no auth yet (public access), set a minimal\n        // auth object so handleParseSession skips session resolution.\n        if (!sessionToken && !req.auth) {\n          req.auth = { isMaster: false };\n        }\n      }\n      next();\n    };\n    // Metadata route must come before the catch-all GET route\n    router.get('/files/:appId/metadata/*filepath', initInfo, Middlewares.handleParseSession, this.metadataHandler);\n    router.get('/files/:appId/*filepath', initInfo, Middlewares.handleParseSession, 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  static _validateFileDownload(req, config) {\n    const isMaster = req.auth?.isMaster;\n    const isMaintenance = req.auth?.isMaintenance;\n    if (isMaster || isMaintenance) {\n      return;\n    }\n    const user = req.auth?.user;\n    const isLinked = user && Parse.AnonymousUtils.isLinked(user);\n    if (!config.fileDownload.enableForAnonymousUser && isLinked) {\n      throw new Parse.Error(\n        Parse.Error.OPERATION_FORBIDDEN,\n        'File download by anonymous user is disabled.'\n      );\n    }\n    if (!config.fileDownload.enableForAuthenticatedUser && !isLinked && user) {\n      throw new Parse.Error(\n        Parse.Error.OPERATION_FORBIDDEN,\n        'File download by authenticated user is disabled.'\n      );\n    }\n    if (!config.fileDownload.enableForPublic && !user) {\n      throw new Parse.Error(\n        Parse.Error.OPERATION_FORBIDDEN,\n        'File download by public is disabled.'\n      );\n    }\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    FilesRouter._validateFileDownload(req, config);\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 = req.auth;\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 isMaster = req.auth.isMaster;\n    const isMaintenance = req.auth.isMaintenance;\n    if (!isMaster && !isMaintenance) {\n      const user = req.auth.user;\n      const isLinked = user && Parse.AnonymousUtils.isLinked(user);\n      if (!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 (!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 (!config.fileUpload.enableForPublic && !user) {\n        next(new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'File upload by public is disabled.'));\n        return;\n      }\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      FilesRouter._validateFileDownload(req, config);\n      const { filesController } = config;\n      let filename = FilesRouter._getFilenameFromParams(req);\n      const file = new Parse.File(filename, { base64: '' });\n      const fileAuth = req.auth;\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/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "parse-server",
3
- "version": "9.8.0-alpha.2",
3
+ "version": "9.8.0-alpha.4",
4
4
  "description": "An express module providing a Parse-compatible API server",
5
5
  "main": "lib/index.js",
6
6
  "repository": {
@@ -41,7 +41,7 @@
41
41
  "jsonwebtoken": "9.0.2",
42
42
  "jwks-rsa": "3.2.0",
43
43
  "ldapjs": "3.0.7",
44
- "lodash": "4.17.23",
44
+ "lodash": "4.18.1",
45
45
  "lru-cache": "11.2.7",
46
46
  "mime": "4.1.0",
47
47
  "mongodb": "7.1.0",
@@ -55,7 +55,7 @@
55
55
  "punycode": "2.3.1",
56
56
  "rate-limit-redis": "4.3.1",
57
57
  "redis": "5.11.0",
58
- "semver": "7.7.2",
58
+ "semver": "7.7.4",
59
59
  "tv4": "1.3.0",
60
60
  "uuid": "13.0.0",
61
61
  "winston": "3.19.0",