parse-server 9.3.0-alpha.8 → 9.3.0-alpha.9

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.
@@ -33,7 +33,7 @@ class FilesAdapter {
33
33
  /** Responsible for storing the file in order to be retrieved later by its filename
34
34
  *
35
35
  * @param {string} filename - the filename to save
36
- * @param {*} data - the buffer of data from the file
36
+ * @param {Buffer|import('stream').Readable} data - the file data as a Buffer, or a Readable stream if the adapter supports streaming (see supportsStreaming)
37
37
  * @param {string} contentType - the supposed contentType
38
38
  * @discussion the contentType can be undefined if the controller was not able to determine it
39
39
  * @param {object} options - (Optional) options to be passed to file adapter (S3 File Adapter Only)
@@ -45,6 +45,16 @@ class FilesAdapter {
45
45
  */
46
46
  createFile(filename, data, contentType, options) {}
47
47
 
48
+ /** Whether this adapter supports receiving Readable streams in createFile().
49
+ * If false (default), streams are buffered to a Buffer before being passed.
50
+ * Override and return true to receive Readable streams directly.
51
+ *
52
+ * @return {boolean}
53
+ */
54
+ get supportsStreaming() {
55
+ return false;
56
+ }
57
+
48
58
  /** Responsible for deleting the specified file
49
59
  *
50
60
  * @param {string} filename - the filename to delete
@@ -116,4 +126,4 @@ function validateFilename(filename) {
116
126
  return null;
117
127
  }
118
128
  var _default = exports.default = FilesAdapter;
119
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfbm9kZSIsIl9pbnRlcm9wUmVxdWlyZURlZmF1bHQiLCJyZXF1aXJlIiwiZSIsIl9fZXNNb2R1bGUiLCJkZWZhdWx0IiwiRmlsZXNBZGFwdGVyIiwiY3JlYXRlRmlsZSIsImZpbGVuYW1lIiwiZGF0YSIsImNvbnRlbnRUeXBlIiwib3B0aW9ucyIsImRlbGV0ZUZpbGUiLCJnZXRGaWxlRGF0YSIsImdldEZpbGVMb2NhdGlvbiIsImNvbmZpZyIsImV4cG9ydHMiLCJ2YWxpZGF0ZUZpbGVuYW1lIiwibGVuZ3RoIiwiUGFyc2UiLCJFcnJvciIsIklOVkFMSURfRklMRV9OQU1FIiwicmVneCIsIm1hdGNoIiwiX2RlZmF1bHQiXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvQWRhcHRlcnMvRmlsZXMvRmlsZXNBZGFwdGVyLmpzIl0sInNvdXJjZXNDb250ZW50IjpbIi8qIGVzbGludC1kaXNhYmxlIHVudXNlZC1pbXBvcnRzL25vLXVudXNlZC12YXJzICovXG4vLyBGaWxlcyBBZGFwdGVyXG4vL1xuLy8gQWxsb3dzIHlvdSB0byBjaGFuZ2UgdGhlIGZpbGUgc3RvcmFnZSBtZWNoYW5pc20uXG4vL1xuLy8gQWRhcHRlciBjbGFzc2VzIG11c3QgaW1wbGVtZW50IHRoZSBmb2xsb3dpbmcgZnVuY3Rpb25zOlxuLy8gKiBjcmVhdGVGaWxlKGZpbGVuYW1lLCBkYXRhLCBjb250ZW50VHlwZSlcbi8vICogZGVsZXRlRmlsZShmaWxlbmFtZSlcbi8vICogZ2V0RmlsZURhdGEoZmlsZW5hbWUpXG4vLyAqIGdldEZpbGVMb2NhdGlvbihjb25maWcsIGZpbGVuYW1lKVxuLy8gQWRhcHRlciBjbGFzc2VzIHNob3VsZCBpbXBsZW1lbnQgdGhlIGZvbGxvd2luZyBmdW5jdGlvbnM6XG4vLyAqIHZhbGlkYXRlRmlsZW5hbWUoZmlsZW5hbWUpXG4vLyAqIGhhbmRsZUZpbGVTdHJlYW0oZmlsZW5hbWUsIHJlcSwgcmVzLCBjb250ZW50VHlwZSlcbi8vXG4vLyBEZWZhdWx0IGlzIEdyaWRGU0J1Y2tldEFkYXB0ZXIsIHdoaWNoIHJlcXVpcmVzIG1vbmdvXG4vLyBhbmQgZm9yIHRoZSBBUEkgc2VydmVyIHRvIGJlIHVzaW5nIHRoZSBEYXRhYmFzZUNvbnRyb2xsZXIgd2l0aCBNb25nb1xuLy8gZGF0YWJhc2UgYWRhcHRlci5cblxuaW1wb3J0IHR5cGUgeyBDb25maWcgfSBmcm9tICcuLi8uLi9Db25maWcnO1xuaW1wb3J0IFBhcnNlIGZyb20gJ3BhcnNlL25vZGUnO1xuLyoqXG4gKiBAaW50ZXJmYWNlXG4gKiBAbWVtYmVyb2YgbW9kdWxlOkFkYXB0ZXJzXG4gKi9cbmV4cG9ydCBjbGFzcyBGaWxlc0FkYXB0ZXIge1xuICAvKiogUmVzcG9uc2libGUgZm9yIHN0b3JpbmcgdGhlIGZpbGUgaW4gb3JkZXIgdG8gYmUgcmV0cmlldmVkIGxhdGVyIGJ5IGl0cyBmaWxlbmFtZVxuICAgKlxuICAgKiBAcGFyYW0ge3N0cmluZ30gZmlsZW5hbWUgLSB0aGUgZmlsZW5hbWUgdG8gc2F2ZVxuICAgKiBAcGFyYW0geyp9IGRhdGEgLSB0aGUgYnVmZmVyIG9mIGRhdGEgZnJvbSB0aGUgZmlsZVxuICAgKiBAcGFyYW0ge3N0cmluZ30gY29udGVudFR5cGUgLSB0aGUgc3VwcG9zZWQgY29udGVudFR5cGVcbiAgICogQGRpc2N1c3Npb24gdGhlIGNvbnRlbnRUeXBlIGNhbiBiZSB1bmRlZmluZWQgaWYgdGhlIGNvbnRyb2xsZXIgd2FzIG5vdCBhYmxlIHRvIGRldGVybWluZSBpdFxuICAgKiBAcGFyYW0ge29iamVjdH0gb3B0aW9ucyAtIChPcHRpb25hbCkgb3B0aW9ucyB0byBiZSBwYXNzZWQgdG8gZmlsZSBhZGFwdGVyIChTMyBGaWxlIEFkYXB0ZXIgT25seSlcbiAgICogLSB0YWdzOiBvYmplY3QgY29udGFpbmluZyBrZXkgdmFsdWUgcGFpcnMgdGhhdCB3aWxsIGJlIHN0b3JlZCB3aXRoIGZpbGVcbiAgICogLSBtZXRhZGF0YTogb2JqZWN0IGNvbnRhaW5pbmcga2V5IHZhbHVlIHBhaXJzIHRoYXQgd2lsbCBiZSBzb3RyZWQgd2l0aCBmaWxlIChodHRwczovL2RvY3MuYXdzLmFtYXpvbi5jb20vQW1hem9uUzMvbGF0ZXN0L3VzZXItZ3VpZGUvYWRkLW9iamVjdC1tZXRhZGF0YS5odG1sKVxuICAgKiBAZGlzY3Vzc2lvbiBvcHRpb25zIGFyZSBub3Qgc3VwcG9ydGVkIGJ5IGFsbCBmaWxlIGFkYXB0ZXJzLiBDaGVjayB0aGUgeW91ciBhZGFwdGVyJ3MgZG9jdW1lbnRhdGlvbiBmb3IgY29tcGF0aWJpbGl0eVxuICAgKlxuICAgKiBAcmV0dXJuIHtQcm9taXNlfSBhIHByb21pc2UgdGhhdCBzaG91bGQgZmFpbCBpZiB0aGUgc3RvcmFnZSBkaWRuJ3Qgc3VjY2VlZFxuICAgKi9cbiAgY3JlYXRlRmlsZShmaWxlbmFtZTogc3RyaW5nLCBkYXRhLCBjb250ZW50VHlwZTogc3RyaW5nLCBvcHRpb25zOiBPYmplY3QpOiBQcm9taXNlIHt9XG5cbiAgLyoqIFJlc3BvbnNpYmxlIGZvciBkZWxldGluZyB0aGUgc3BlY2lmaWVkIGZpbGVcbiAgICpcbiAgICogQHBhcmFtIHtzdHJpbmd9IGZpbGVuYW1lIC0gdGhlIGZpbGVuYW1lIHRvIGRlbGV0ZVxuICAgKlxuICAgKiBAcmV0dXJuIHtQcm9taXNlfSBhIHByb21pc2UgdGhhdCBzaG91bGQgZmFpbCBpZiB0aGUgZGVsZXRpb24gZGlkbid0IHN1Y2NlZWRcbiAgICovXG4gIGRlbGV0ZUZpbGUoZmlsZW5hbWU6IHN0cmluZyk6IFByb21pc2Uge31cblxuICAvKiogUmVzcG9uc2libGUgZm9yIHJldHJpZXZpbmcgdGhlIGRhdGEgb2YgdGhlIHNwZWNpZmllZCBmaWxlXG4gICAqXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBmaWxlbmFtZSAtIHRoZSBuYW1lIG9mIGZpbGUgdG8gcmV0cmlldmVcbiAgICpcbiAgICogQHJldHVybiB7UHJvbWlzZX0gYSBwcm9taXNlIHRoYXQgc2hvdWxkIHBhc3Mgd2l0aCB0aGUgZmlsZSBkYXRhIG9yIGZhaWwgb24gZXJyb3JcbiAgICovXG4gIGdldEZpbGVEYXRhKGZpbGVuYW1lOiBzdHJpbmcpOiBQcm9taXNlPGFueT4ge31cblxuICAvKiogUmV0dXJucyBhbiBhYnNvbHV0ZSBVUkwgd2hlcmUgdGhlIGZpbGUgY2FuIGJlIGFjY2Vzc2VkXG4gICAqXG4gICAqIEBwYXJhbSB7Q29uZmlnfSBjb25maWcgLSBzZXJ2ZXIgY29uZmlndXJhdGlvblxuICAgKiBAcGFyYW0ge3N0cmluZ30gZmlsZW5hbWVcbiAgICpcbiAgICogQHJldHVybiB7c3RyaW5nIHwgUHJvbWlzZTxzdHJpbmc+fSBBYnNvbHV0ZSBVUkxcbiAgICovXG4gIGdldEZpbGVMb2NhdGlvbihjb25maWc6IENvbmZpZywgZmlsZW5hbWU6IHN0cmluZyk6IHN0cmluZyB8IFByb21pc2U8c3RyaW5nPiB7fVxuXG4gIC8qKiBWYWxpZGF0ZSBhIGZpbGVuYW1lIGZvciB0aGlzIGFkYXB0ZXIgdHlwZVxuICAgKlxuICAgKiBAcGFyYW0ge3N0cmluZ30gZmlsZW5hbWVcbiAgICpcbiAgICogQHJldHVybnMge251bGx8UGFyc2UuRXJyb3J9IG51bGwgaWYgdGhlcmUgYXJlIG5vIGVycm9yc1xuICAgKi9cbiAgLy8gdmFsaWRhdGVGaWxlbmFtZShmaWxlbmFtZTogc3RyaW5nKTogP1BhcnNlLkVycm9yIHt9XG5cbiAgLyoqIEhhbmRsZXMgQnl0ZS1SYW5nZSBSZXF1ZXN0cyBmb3IgU3RyZWFtaW5nXG4gICAqXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBmaWxlbmFtZVxuICAgKiBAcGFyYW0ge29iamVjdH0gcmVxXG4gICAqIEBwYXJhbSB7b2JqZWN0fSByZXNcbiAgICogQHBhcmFtIHtzdHJpbmd9IGNvbnRlbnRUeXBlXG4gICAqXG4gICAqIEByZXR1cm5zIHtQcm9taXNlfSBEYXRhIGZvciBieXRlIHJhbmdlXG4gICAqL1xuICAvLyBoYW5kbGVGaWxlU3RyZWFtKGZpbGVuYW1lOiBzdHJpbmcsIHJlczogYW55LCByZXE6IGFueSwgY29udGVudFR5cGU6IHN0cmluZyk6IFByb21pc2VcblxuICAvKiogUmVzcG9uc2libGUgZm9yIHJldHJpZXZpbmcgbWV0YWRhdGEgYW5kIHRhZ3NcbiAgICpcbiAgICogQHBhcmFtIHtzdHJpbmd9IGZpbGVuYW1lIC0gdGhlIGZpbGVuYW1lIHRvIHJldHJpZXZlIG1ldGFkYXRhXG4gICAqXG4gICAqIEByZXR1cm4ge1Byb21pc2V9IGEgcHJvbWlzZSB0aGF0IHNob3VsZCBwYXNzIHdpdGggbWV0YWRhdGFcbiAgICovXG4gIC8vIGdldE1ldGFkYXRhKGZpbGVuYW1lOiBzdHJpbmcpOiBQcm9taXNlPGFueT4ge31cbn1cblxuLyoqXG4gKiBTaW1wbGUgZmlsZW5hbWUgdmFsaWRhdGlvblxuICpcbiAqIEBwYXJhbSBmaWxlbmFtZVxuICogQHJldHVybnMge251bGx8UGFyc2UuRXJyb3J9XG4gKi9cbmV4cG9ydCBmdW5jdGlvbiB2YWxpZGF0ZUZpbGVuYW1lKGZpbGVuYW1lKTogP1BhcnNlLkVycm9yIHtcbiAgaWYgKGZpbGVuYW1lLmxlbmd0aCA+IDEyOCkge1xuICAgIHJldHVybiBuZXcgUGFyc2UuRXJyb3IoUGFyc2UuRXJyb3IuSU5WQUxJRF9GSUxFX05BTUUsICdGaWxlbmFtZSB0b28gbG9uZy4nKTtcbiAgfVxuXG4gIGNvbnN0IHJlZ3ggPSAvXltfYS16QS1aMC05XVthLXpBLVowLTlALiB+Xy1dKiQvO1xuICBpZiAoIWZpbGVuYW1lLm1hdGNoKHJlZ3gpKSB7XG4gICAgcmV0dXJuIG5ldyBQYXJzZS5FcnJvcihQYXJzZS5FcnJvci5JTlZBTElEX0ZJTEVfTkFNRSwgJ0ZpbGVuYW1lIGNvbnRhaW5zIGludmFsaWQgY2hhcmFjdGVycy4nKTtcbiAgfVxuICByZXR1cm4gbnVsbDtcbn1cblxuZXhwb3J0IGRlZmF1bHQgRmlsZXNBZGFwdGVyO1xuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7O0FBbUJBLElBQUFBLEtBQUEsR0FBQUMsc0JBQUEsQ0FBQUMsT0FBQTtBQUErQixTQUFBRCx1QkFBQUUsQ0FBQSxXQUFBQSxDQUFBLElBQUFBLENBQUEsQ0FBQUMsVUFBQSxHQUFBRCxDQUFBLEtBQUFFLE9BQUEsRUFBQUYsQ0FBQTtBQW5CL0I7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFJQTtBQUNBO0FBQ0E7QUFDQTtBQUNPLE1BQU1HLFlBQVksQ0FBQztFQUN4QjtBQUNGO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtFQUNFQyxVQUFVQSxDQUFDQyxRQUFnQixFQUFFQyxJQUFJLEVBQUVDLFdBQW1CLEVBQUVDLE9BQWUsRUFBVyxDQUFDOztFQUVuRjtBQUNGO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7RUFDRUMsVUFBVUEsQ0FBQ0osUUFBZ0IsRUFBVyxDQUFDOztFQUV2QztBQUNGO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7RUFDRUssV0FBV0EsQ0FBQ0wsUUFBZ0IsRUFBZ0IsQ0FBQzs7RUFFN0M7QUFDRjtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7RUFDRU0sZUFBZUEsQ0FBQ0MsTUFBYyxFQUFFUCxRQUFnQixFQUE0QixDQUFDOztFQUU3RTtBQUNGO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7RUFDRTs7RUFFQTtBQUNGO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7RUFDRTs7RUFFQTtBQUNGO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7RUFDRTtBQUNGOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUxBUSxPQUFBLENBQUFWLFlBQUEsR0FBQUEsWUFBQTtBQU1PLFNBQVNXLGdCQUFnQkEsQ0FBQ1QsUUFBUSxFQUFnQjtFQUN2RCxJQUFJQSxRQUFRLENBQUNVLE1BQU0sR0FBRyxHQUFHLEVBQUU7SUFDekIsT0FBTyxJQUFJQyxhQUFLLENBQUNDLEtBQUssQ0FBQ0QsYUFBSyxDQUFDQyxLQUFLLENBQUNDLGlCQUFpQixFQUFFLG9CQUFvQixDQUFDO0VBQzdFO0VBRUEsTUFBTUMsSUFBSSxHQUFHLGtDQUFrQztFQUMvQyxJQUFJLENBQUNkLFFBQVEsQ0FBQ2UsS0FBSyxDQUFDRCxJQUFJLENBQUMsRUFBRTtJQUN6QixPQUFPLElBQUlILGFBQUssQ0FBQ0MsS0FBSyxDQUFDRCxhQUFLLENBQUNDLEtBQUssQ0FBQ0MsaUJBQWlCLEVBQUUsdUNBQXVDLENBQUM7RUFDaEc7RUFDQSxPQUFPLElBQUk7QUFDYjtBQUFDLElBQUFHLFFBQUEsR0FBQVIsT0FBQSxDQUFBWCxPQUFBLEdBRWNDLFlBQVkiLCJpZ25vcmVMaXN0IjpbXX0=
129
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_node","_interopRequireDefault","require","e","__esModule","default","FilesAdapter","createFile","filename","data","contentType","options","supportsStreaming","deleteFile","getFileData","getFileLocation","config","exports","validateFilename","length","Parse","Error","INVALID_FILE_NAME","regx","match","_default"],"sources":["../../../src/Adapters/Files/FilesAdapter.js"],"sourcesContent":["/* eslint-disable unused-imports/no-unused-vars */\n// Files Adapter\n//\n// Allows you to change the file storage mechanism.\n//\n// Adapter classes must implement the following functions:\n// * createFile(filename, data, contentType)\n// * deleteFile(filename)\n// * getFileData(filename)\n// * getFileLocation(config, filename)\n// Adapter classes should implement the following functions:\n// * validateFilename(filename)\n// * handleFileStream(filename, req, res, contentType)\n//\n// Default is GridFSBucketAdapter, which requires mongo\n// and for the API server to be using the DatabaseController with Mongo\n// database adapter.\n\nimport type { Config } from '../../Config';\nimport Parse from 'parse/node';\n/**\n * @interface\n * @memberof module:Adapters\n */\nexport class FilesAdapter {\n  /** Responsible for storing the file in order to be retrieved later by its filename\n   *\n   * @param {string} filename - the filename to save\n   * @param {Buffer|import('stream').Readable} data - the file data as a Buffer, or a Readable stream if the adapter supports streaming (see supportsStreaming)\n   * @param {string} contentType - the supposed contentType\n   * @discussion the contentType can be undefined if the controller was not able to determine it\n   * @param {object} options - (Optional) options to be passed to file adapter (S3 File Adapter Only)\n   * - tags: object containing key value pairs that will be stored with file\n   * - metadata: object containing key value pairs that will be sotred with file (https://docs.aws.amazon.com/AmazonS3/latest/user-guide/add-object-metadata.html)\n   * @discussion options are not supported by all file adapters. Check the your adapter's documentation for compatibility\n   *\n   * @return {Promise} a promise that should fail if the storage didn't succeed\n   */\n  createFile(filename: string, data, contentType: string, options: Object): Promise {}\n\n  /** Whether this adapter supports receiving Readable streams in createFile().\n   * If false (default), streams are buffered to a Buffer before being passed.\n   * Override and return true to receive Readable streams directly.\n   *\n   * @return {boolean}\n   */\n  get supportsStreaming() {\n    return false;\n  }\n\n  /** Responsible for deleting the specified file\n   *\n   * @param {string} filename - the filename to delete\n   *\n   * @return {Promise} a promise that should fail if the deletion didn't succeed\n   */\n  deleteFile(filename: string): Promise {}\n\n  /** Responsible for retrieving the data of the specified file\n   *\n   * @param {string} filename - the name of file to retrieve\n   *\n   * @return {Promise} a promise that should pass with the file data or fail on error\n   */\n  getFileData(filename: string): Promise<any> {}\n\n  /** Returns an absolute URL where the file can be accessed\n   *\n   * @param {Config} config - server configuration\n   * @param {string} filename\n   *\n   * @return {string | Promise<string>} Absolute URL\n   */\n  getFileLocation(config: Config, filename: string): string | Promise<string> {}\n\n  /** Validate a filename for this adapter type\n   *\n   * @param {string} filename\n   *\n   * @returns {null|Parse.Error} null if there are no errors\n   */\n  // validateFilename(filename: string): ?Parse.Error {}\n\n  /** Handles Byte-Range Requests for Streaming\n   *\n   * @param {string} filename\n   * @param {object} req\n   * @param {object} res\n   * @param {string} contentType\n   *\n   * @returns {Promise} Data for byte range\n   */\n  // handleFileStream(filename: string, res: any, req: any, contentType: string): Promise\n\n  /** Responsible for retrieving metadata and tags\n   *\n   * @param {string} filename - the filename to retrieve metadata\n   *\n   * @return {Promise} a promise that should pass with metadata\n   */\n  // getMetadata(filename: string): Promise<any> {}\n}\n\n/**\n * Simple filename validation\n *\n * @param filename\n * @returns {null|Parse.Error}\n */\nexport function validateFilename(filename): ?Parse.Error {\n  if (filename.length > 128) {\n    return new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Filename too long.');\n  }\n\n  const regx = /^[_a-zA-Z0-9][a-zA-Z0-9@. ~_-]*$/;\n  if (!filename.match(regx)) {\n    return new Parse.Error(Parse.Error.INVALID_FILE_NAME, 'Filename contains invalid characters.');\n  }\n  return null;\n}\n\nexport default FilesAdapter;\n"],"mappings":";;;;;;;AAmBA,IAAAA,KAAA,GAAAC,sBAAA,CAAAC,OAAA;AAA+B,SAAAD,uBAAAE,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAnB/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAIA;AACA;AACA;AACA;AACO,MAAMG,YAAY,CAAC;EACxB;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACEC,UAAUA,CAACC,QAAgB,EAAEC,IAAI,EAAEC,WAAmB,EAAEC,OAAe,EAAW,CAAC;;EAEnF;AACF;AACA;AACA;AACA;AACA;EACE,IAAIC,iBAAiBA,CAAA,EAAG;IACtB,OAAO,KAAK;EACd;;EAEA;AACF;AACA;AACA;AACA;AACA;EACEC,UAAUA,CAACL,QAAgB,EAAW,CAAC;;EAEvC;AACF;AACA;AACA;AACA;AACA;EACEM,WAAWA,CAACN,QAAgB,EAAgB,CAAC;;EAE7C;AACF;AACA;AACA;AACA;AACA;AACA;EACEO,eAAeA,CAACC,MAAc,EAAER,QAAgB,EAA4B,CAAC;;EAE7E;AACF;AACA;AACA;AACA;AACA;EACE;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACE;;EAEA;AACF;AACA;AACA;AACA;AACA;EACE;AACF;;AAEA;AACA;AACA;AACA;AACA;AACA;AALAS,OAAA,CAAAX,YAAA,GAAAA,YAAA;AAMO,SAASY,gBAAgBA,CAACV,QAAQ,EAAgB;EACvD,IAAIA,QAAQ,CAACW,MAAM,GAAG,GAAG,EAAE;IACzB,OAAO,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACC,iBAAiB,EAAE,oBAAoB,CAAC;EAC7E;EAEA,MAAMC,IAAI,GAAG,kCAAkC;EAC/C,IAAI,CAACf,QAAQ,CAACgB,KAAK,CAACD,IAAI,CAAC,EAAE;IACzB,OAAO,IAAIH,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACC,iBAAiB,EAAE,uCAAuC,CAAC;EAChG;EACA,OAAO,IAAI;AACb;AAAC,IAAAG,QAAA,GAAAR,OAAA,CAAAZ,OAAA,GAEcC,YAAY","ignoreList":[]}
@@ -34,6 +34,9 @@ class GridFSBucketAdapter extends _FilesAdapter.FilesAdapter {
34
34
  }
35
35
  this._mongoOptions = _mongoOptions;
36
36
  }
37
+ get supportsStreaming() {
38
+ return true;
39
+ }
37
40
  _connect() {
38
41
  if (!this._connectionPromise) {
39
42
  // Only use driverInfo if clientMetadata option is set
@@ -64,6 +67,31 @@ class GridFSBucketAdapter extends _FilesAdapter.FilesAdapter {
64
67
  const stream = await bucket.openUploadStream(filename, {
65
68
  metadata: options.metadata
66
69
  });
70
+
71
+ // If data is a stream and encryption is enabled, buffer first
72
+ // (AES-256-GCM needs complete data for format: [encrypted][IV][authTag])
73
+ if (typeof data?.pipe === 'function' && this._encryptionKey !== null) {
74
+ data = await new Promise((resolve, reject) => {
75
+ const chunks = [];
76
+ data.on('data', chunk => chunks.push(chunk));
77
+ data.on('end', () => resolve(Buffer.concat(chunks)));
78
+ data.on('error', reject);
79
+ });
80
+ }
81
+ if (typeof data?.pipe === 'function') {
82
+ // Pipe readable stream directly into GridFS upload stream
83
+ return new Promise((resolve, reject) => {
84
+ data.pipe(stream);
85
+ stream.on('finish', resolve);
86
+ stream.on('error', reject);
87
+ data.on('error', err => {
88
+ stream.destroy(err);
89
+ reject(err);
90
+ });
91
+ });
92
+ }
93
+
94
+ // Buffer path (existing behavior)
67
95
  if (this._encryptionKey !== null) {
68
96
  try {
69
97
  const iv = crypto.randomBytes(16);
@@ -239,4 +267,4 @@ class GridFSBucketAdapter extends _FilesAdapter.FilesAdapter {
239
267
  }
240
268
  exports.GridFSBucketAdapter = GridFSBucketAdapter;
241
269
  var _default = exports.default = GridFSBucketAdapter;
242
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_mongodb","require","_FilesAdapter","_defaults","_interopRequireWildcard","e","t","WeakMap","r","n","__esModule","o","i","f","__proto__","default","has","get","set","hasOwnProperty","call","Object","defineProperty","getOwnPropertyDescriptor","crypto","GridFSBucketAdapter","FilesAdapter","constructor","mongoDatabaseURI","defaults","DefaultMongoURI","mongoOptions","encryptionKey","undefined","_databaseURI","_algorithm","_encryptionKey","createHash","update","String","digest","substring","defaultMongoOptions","_mongoOptions","assign","_clientMetadata","clientMetadata","key","ParseServerDatabaseOptions","_connect","_connectionPromise","options","driverInfo","name","version","MongoClient","connect","then","client","_client","db","s","dbName","_getBucket","database","GridFSBucket","createFile","filename","data","contentType","bucket","stream","openUploadStream","metadata","iv","randomBytes","cipher","createCipheriv","encryptedResult","Buffer","concat","final","getAuthTag","write","err","Promise","resolve","reject","end","on","deleteFile","documents","find","toArray","length","Error","all","map","doc","delete","_id","getFileData","openDownloadStreamByName","read","chunks","push","authTagLocation","ivLocation","authTag","slice","encrypted","decipher","createDecipheriv","setAuthTag","decrypted","rotateEncryptionKey","fileNames","oldKeyFileAdapter","oldKey","fileNamesIterator","forEach","file","fileNamesNotRotated","fileNamesRotated","fileName","plainTextData","filter","value","rotated","notRotated","getFileLocation","config","mount","applicationId","encodeURIComponent","getMetadata","files","handleFileStream","req","res","parts","replace","split","partialstart","partialend","fileLength","fileStart","parseInt","fileEnd","start","Math","min","max","isNaN","status","header","chunk","send","message","handleShutdown","close","validateFilename","exports","_default"],"sources":["../../../src/Adapters/Files/GridFSBucketAdapter.js"],"sourcesContent":["/**\n GridFSBucketAdapter\n Stores files in Mongo using GridFS\n Requires the database adapter to be based on mongoclient\n\n @flow weak\n */\n\n// @flow-disable-next\nimport { MongoClient, GridFSBucket, Db } from 'mongodb';\nimport { FilesAdapter, validateFilename } from './FilesAdapter';\nimport defaults, { ParseServerDatabaseOptions } from '../../defaults';\nconst crypto = require('crypto');\n\nexport class GridFSBucketAdapter extends FilesAdapter {\n  _databaseURI: string;\n  _connectionPromise: Promise<Db>;\n  _mongoOptions: Object;\n  _algorithm: string;\n  _clientMetadata: ?{ name: string, version: string };\n\n  constructor(\n    mongoDatabaseURI = defaults.DefaultMongoURI,\n    mongoOptions = {},\n    encryptionKey = undefined\n  ) {\n    super();\n    this._databaseURI = mongoDatabaseURI;\n    this._algorithm = 'aes-256-gcm';\n    this._encryptionKey =\n      encryptionKey !== undefined\n        ? crypto\n          .createHash('sha256')\n          .update(String(encryptionKey))\n          .digest('base64')\n          .substring(0, 32)\n        : null;\n    const defaultMongoOptions = {};\n    const _mongoOptions = Object.assign(defaultMongoOptions, mongoOptions);\n    this._clientMetadata = mongoOptions.clientMetadata;\n    // Remove Parse Server-specific options that should not be passed to MongoDB client\n    for (const key of ParseServerDatabaseOptions) {\n      delete _mongoOptions[key];\n    }\n    this._mongoOptions = _mongoOptions;\n  }\n\n  _connect() {\n    if (!this._connectionPromise) {\n      // Only use driverInfo if clientMetadata option is set\n      const options = { ...this._mongoOptions };\n      if (this._clientMetadata) {\n        options.driverInfo = {\n          name: this._clientMetadata.name,\n          version: this._clientMetadata.version\n        };\n      }\n\n      this._connectionPromise = MongoClient.connect(this._databaseURI, options).then(\n        client => {\n          this._client = client;\n          return client.db(client.s.options.dbName);\n        }\n      );\n    }\n    return this._connectionPromise;\n  }\n\n  _getBucket() {\n    return this._connect().then(database => new GridFSBucket(database));\n  }\n\n  // For a given config object, filename, and data, store a file\n  // Returns a promise\n  async createFile(filename: string, data, contentType, options = {}) {\n    const bucket = await this._getBucket();\n    const stream = await bucket.openUploadStream(filename, {\n      metadata: options.metadata,\n    });\n    if (this._encryptionKey !== null) {\n      try {\n        const iv = crypto.randomBytes(16);\n        const cipher = crypto.createCipheriv(this._algorithm, this._encryptionKey, iv);\n        const encryptedResult = Buffer.concat([\n          cipher.update(data),\n          cipher.final(),\n          iv,\n          cipher.getAuthTag(),\n        ]);\n        await stream.write(encryptedResult);\n      } catch (err) {\n        return new Promise((resolve, reject) => {\n          return reject(err);\n        });\n      }\n    } else {\n      await stream.write(data);\n    }\n    stream.end();\n    return new Promise((resolve, reject) => {\n      stream.on('finish', resolve);\n      stream.on('error', reject);\n    });\n  }\n\n  async deleteFile(filename: string) {\n    const bucket = await this._getBucket();\n    const documents = await bucket.find({ filename }).toArray();\n    if (documents.length === 0) {\n      throw new Error('FileNotFound');\n    }\n    return Promise.all(\n      documents.map(doc => {\n        return bucket.delete(doc._id);\n      })\n    );\n  }\n\n  async getFileData(filename: string) {\n    const bucket = await this._getBucket();\n    const stream = bucket.openDownloadStreamByName(filename);\n    stream.read();\n    return new Promise((resolve, reject) => {\n      const chunks = [];\n      stream.on('data', data => {\n        chunks.push(data);\n      });\n      stream.on('end', () => {\n        const data = Buffer.concat(chunks);\n        if (this._encryptionKey !== null) {\n          try {\n            const authTagLocation = data.length - 16;\n            const ivLocation = data.length - 32;\n            const authTag = data.slice(authTagLocation);\n            const iv = data.slice(ivLocation, authTagLocation);\n            const encrypted = data.slice(0, ivLocation);\n            const decipher = crypto.createDecipheriv(this._algorithm, this._encryptionKey, iv);\n            decipher.setAuthTag(authTag);\n            const decrypted = Buffer.concat([decipher.update(encrypted), decipher.final()]);\n            return resolve(decrypted);\n          } catch (err) {\n            return reject(err);\n          }\n        }\n        resolve(data);\n      });\n      stream.on('error', err => {\n        reject(err);\n      });\n    });\n  }\n\n  async rotateEncryptionKey(options = {}) {\n    let fileNames = [];\n    let oldKeyFileAdapter = {};\n    const bucket = await this._getBucket();\n    if (options.oldKey !== undefined) {\n      oldKeyFileAdapter = new GridFSBucketAdapter(\n        this._databaseURI,\n        this._mongoOptions,\n        options.oldKey\n      );\n    } else {\n      oldKeyFileAdapter = new GridFSBucketAdapter(this._databaseURI, this._mongoOptions);\n    }\n    if (options.fileNames !== undefined) {\n      fileNames = options.fileNames;\n    } else {\n      const fileNamesIterator = await bucket.find().toArray();\n      fileNamesIterator.forEach(file => {\n        fileNames.push(file.filename);\n      });\n    }\n    let fileNamesNotRotated = fileNames;\n    const fileNamesRotated = [];\n    for (const fileName of fileNames) {\n      try {\n        const plainTextData = await oldKeyFileAdapter.getFileData(fileName);\n        // Overwrite file with data encrypted with new key\n        await this.createFile(fileName, plainTextData);\n        fileNamesRotated.push(fileName);\n        fileNamesNotRotated = fileNamesNotRotated.filter(function (value) {\n          return value !== fileName;\n        });\n      } catch {\n        continue;\n      }\n    }\n    return { rotated: fileNamesRotated, notRotated: fileNamesNotRotated };\n  }\n\n  getFileLocation(config, filename) {\n    return config.mount + '/files/' + config.applicationId + '/' + encodeURIComponent(filename);\n  }\n\n  async getMetadata(filename) {\n    const bucket = await this._getBucket();\n    const files = await bucket.find({ filename }).toArray();\n    if (files.length === 0) {\n      return {};\n    }\n    const { metadata } = files[0];\n    return { metadata };\n  }\n\n  async handleFileStream(filename: string, req, res, contentType) {\n    const bucket = await this._getBucket();\n    const files = await bucket.find({ filename }).toArray();\n    if (files.length === 0) {\n      throw new Error('FileNotFound');\n    }\n    const parts = req\n      .get('Range')\n      .replace(/bytes=/, '')\n      .split('-');\n    const partialstart = parts[0];\n    const partialend = parts[1];\n\n    const fileLength = files[0].length;\n    const fileStart = parseInt(partialstart, 10);\n    const fileEnd = partialend ? parseInt(partialend, 10) : fileLength;\n\n    let start = Math.min(fileStart || 0, fileEnd, fileLength);\n    let end = Math.max(fileStart || 0, fileEnd) + 1 || fileLength;\n    if (isNaN(fileStart)) {\n      start = fileLength - end + 1;\n      end = fileLength;\n    }\n    end = Math.min(end, fileLength);\n    start = Math.max(start, 0);\n\n    res.status(206);\n    res.header('Accept-Ranges', 'bytes');\n    res.header('Content-Length', end - start);\n    res.header('Content-Range', 'bytes ' + start + '-' + end + '/' + fileLength);\n    res.header('Content-Type', contentType);\n    const stream = bucket.openDownloadStreamByName(filename);\n    stream.start(start);\n    if (end) {\n      stream.end(end);\n    }\n    stream.on('data', chunk => {\n      res.write(chunk);\n    });\n    stream.on('error', e => {\n      res.status(404);\n      res.send(e.message);\n    });\n    stream.on('end', () => {\n      res.end();\n    });\n  }\n\n  handleShutdown() {\n    if (!this._client) {\n      return Promise.resolve();\n    }\n    return this._client.close(false);\n  }\n\n  validateFilename(filename) {\n    return validateFilename(filename);\n  }\n}\n\nexport default GridFSBucketAdapter;\n"],"mappings":"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","ignoreList":[]}
270
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_mongodb","require","_FilesAdapter","_defaults","_interopRequireWildcard","e","t","WeakMap","r","n","__esModule","o","i","f","__proto__","default","has","get","set","hasOwnProperty","call","Object","defineProperty","getOwnPropertyDescriptor","crypto","GridFSBucketAdapter","FilesAdapter","constructor","mongoDatabaseURI","defaults","DefaultMongoURI","mongoOptions","encryptionKey","undefined","_databaseURI","_algorithm","_encryptionKey","createHash","update","String","digest","substring","defaultMongoOptions","_mongoOptions","assign","_clientMetadata","clientMetadata","key","ParseServerDatabaseOptions","supportsStreaming","_connect","_connectionPromise","options","driverInfo","name","version","MongoClient","connect","then","client","_client","db","s","dbName","_getBucket","database","GridFSBucket","createFile","filename","data","contentType","bucket","stream","openUploadStream","metadata","pipe","Promise","resolve","reject","chunks","on","chunk","push","Buffer","concat","err","destroy","iv","randomBytes","cipher","createCipheriv","encryptedResult","final","getAuthTag","write","end","deleteFile","documents","find","toArray","length","Error","all","map","doc","delete","_id","getFileData","openDownloadStreamByName","read","authTagLocation","ivLocation","authTag","slice","encrypted","decipher","createDecipheriv","setAuthTag","decrypted","rotateEncryptionKey","fileNames","oldKeyFileAdapter","oldKey","fileNamesIterator","forEach","file","fileNamesNotRotated","fileNamesRotated","fileName","plainTextData","filter","value","rotated","notRotated","getFileLocation","config","mount","applicationId","encodeURIComponent","getMetadata","files","handleFileStream","req","res","parts","replace","split","partialstart","partialend","fileLength","fileStart","parseInt","fileEnd","start","Math","min","max","isNaN","status","header","send","message","handleShutdown","close","validateFilename","exports","_default"],"sources":["../../../src/Adapters/Files/GridFSBucketAdapter.js"],"sourcesContent":["/**\n GridFSBucketAdapter\n Stores files in Mongo using GridFS\n Requires the database adapter to be based on mongoclient\n\n @flow weak\n */\n\n// @flow-disable-next\nimport { MongoClient, GridFSBucket, Db } from 'mongodb';\nimport { FilesAdapter, validateFilename } from './FilesAdapter';\nimport defaults, { ParseServerDatabaseOptions } from '../../defaults';\nconst crypto = require('crypto');\n\nexport class GridFSBucketAdapter extends FilesAdapter {\n  _databaseURI: string;\n  _connectionPromise: Promise<Db>;\n  _mongoOptions: Object;\n  _algorithm: string;\n  _clientMetadata: ?{ name: string, version: string };\n\n  constructor(\n    mongoDatabaseURI = defaults.DefaultMongoURI,\n    mongoOptions = {},\n    encryptionKey = undefined\n  ) {\n    super();\n    this._databaseURI = mongoDatabaseURI;\n    this._algorithm = 'aes-256-gcm';\n    this._encryptionKey =\n      encryptionKey !== undefined\n        ? crypto\n          .createHash('sha256')\n          .update(String(encryptionKey))\n          .digest('base64')\n          .substring(0, 32)\n        : null;\n    const defaultMongoOptions = {};\n    const _mongoOptions = Object.assign(defaultMongoOptions, mongoOptions);\n    this._clientMetadata = mongoOptions.clientMetadata;\n    // Remove Parse Server-specific options that should not be passed to MongoDB client\n    for (const key of ParseServerDatabaseOptions) {\n      delete _mongoOptions[key];\n    }\n    this._mongoOptions = _mongoOptions;\n  }\n\n  get supportsStreaming() {\n    return true;\n  }\n\n  _connect() {\n    if (!this._connectionPromise) {\n      // Only use driverInfo if clientMetadata option is set\n      const options = { ...this._mongoOptions };\n      if (this._clientMetadata) {\n        options.driverInfo = {\n          name: this._clientMetadata.name,\n          version: this._clientMetadata.version\n        };\n      }\n\n      this._connectionPromise = MongoClient.connect(this._databaseURI, options).then(\n        client => {\n          this._client = client;\n          return client.db(client.s.options.dbName);\n        }\n      );\n    }\n    return this._connectionPromise;\n  }\n\n  _getBucket() {\n    return this._connect().then(database => new GridFSBucket(database));\n  }\n\n  // For a given config object, filename, and data, store a file\n  // Returns a promise\n  async createFile(filename: string, data, contentType, options = {}) {\n    const bucket = await this._getBucket();\n    const stream = await bucket.openUploadStream(filename, {\n      metadata: options.metadata,\n    });\n\n    // If data is a stream and encryption is enabled, buffer first\n    // (AES-256-GCM needs complete data for format: [encrypted][IV][authTag])\n    if (typeof data?.pipe === 'function' && this._encryptionKey !== null) {\n      data = await new Promise((resolve, reject) => {\n        const chunks = [];\n        data.on('data', chunk => chunks.push(chunk));\n        data.on('end', () => resolve(Buffer.concat(chunks)));\n        data.on('error', reject);\n      });\n    }\n\n    if (typeof data?.pipe === 'function') {\n      // Pipe readable stream directly into GridFS upload stream\n      return new Promise((resolve, reject) => {\n        data.pipe(stream);\n        stream.on('finish', resolve);\n        stream.on('error', reject);\n        data.on('error', (err) => {\n          stream.destroy(err);\n          reject(err);\n        });\n      });\n    }\n\n    // Buffer path (existing behavior)\n    if (this._encryptionKey !== null) {\n      try {\n        const iv = crypto.randomBytes(16);\n        const cipher = crypto.createCipheriv(this._algorithm, this._encryptionKey, iv);\n        const encryptedResult = Buffer.concat([\n          cipher.update(data),\n          cipher.final(),\n          iv,\n          cipher.getAuthTag(),\n        ]);\n        await stream.write(encryptedResult);\n      } catch (err) {\n        return new Promise((resolve, reject) => {\n          return reject(err);\n        });\n      }\n    } else {\n      await stream.write(data);\n    }\n    stream.end();\n    return new Promise((resolve, reject) => {\n      stream.on('finish', resolve);\n      stream.on('error', reject);\n    });\n  }\n\n  async deleteFile(filename: string) {\n    const bucket = await this._getBucket();\n    const documents = await bucket.find({ filename }).toArray();\n    if (documents.length === 0) {\n      throw new Error('FileNotFound');\n    }\n    return Promise.all(\n      documents.map(doc => {\n        return bucket.delete(doc._id);\n      })\n    );\n  }\n\n  async getFileData(filename: string) {\n    const bucket = await this._getBucket();\n    const stream = bucket.openDownloadStreamByName(filename);\n    stream.read();\n    return new Promise((resolve, reject) => {\n      const chunks = [];\n      stream.on('data', data => {\n        chunks.push(data);\n      });\n      stream.on('end', () => {\n        const data = Buffer.concat(chunks);\n        if (this._encryptionKey !== null) {\n          try {\n            const authTagLocation = data.length - 16;\n            const ivLocation = data.length - 32;\n            const authTag = data.slice(authTagLocation);\n            const iv = data.slice(ivLocation, authTagLocation);\n            const encrypted = data.slice(0, ivLocation);\n            const decipher = crypto.createDecipheriv(this._algorithm, this._encryptionKey, iv);\n            decipher.setAuthTag(authTag);\n            const decrypted = Buffer.concat([decipher.update(encrypted), decipher.final()]);\n            return resolve(decrypted);\n          } catch (err) {\n            return reject(err);\n          }\n        }\n        resolve(data);\n      });\n      stream.on('error', err => {\n        reject(err);\n      });\n    });\n  }\n\n  async rotateEncryptionKey(options = {}) {\n    let fileNames = [];\n    let oldKeyFileAdapter = {};\n    const bucket = await this._getBucket();\n    if (options.oldKey !== undefined) {\n      oldKeyFileAdapter = new GridFSBucketAdapter(\n        this._databaseURI,\n        this._mongoOptions,\n        options.oldKey\n      );\n    } else {\n      oldKeyFileAdapter = new GridFSBucketAdapter(this._databaseURI, this._mongoOptions);\n    }\n    if (options.fileNames !== undefined) {\n      fileNames = options.fileNames;\n    } else {\n      const fileNamesIterator = await bucket.find().toArray();\n      fileNamesIterator.forEach(file => {\n        fileNames.push(file.filename);\n      });\n    }\n    let fileNamesNotRotated = fileNames;\n    const fileNamesRotated = [];\n    for (const fileName of fileNames) {\n      try {\n        const plainTextData = await oldKeyFileAdapter.getFileData(fileName);\n        // Overwrite file with data encrypted with new key\n        await this.createFile(fileName, plainTextData);\n        fileNamesRotated.push(fileName);\n        fileNamesNotRotated = fileNamesNotRotated.filter(function (value) {\n          return value !== fileName;\n        });\n      } catch {\n        continue;\n      }\n    }\n    return { rotated: fileNamesRotated, notRotated: fileNamesNotRotated };\n  }\n\n  getFileLocation(config, filename) {\n    return config.mount + '/files/' + config.applicationId + '/' + encodeURIComponent(filename);\n  }\n\n  async getMetadata(filename) {\n    const bucket = await this._getBucket();\n    const files = await bucket.find({ filename }).toArray();\n    if (files.length === 0) {\n      return {};\n    }\n    const { metadata } = files[0];\n    return { metadata };\n  }\n\n  async handleFileStream(filename: string, req, res, contentType) {\n    const bucket = await this._getBucket();\n    const files = await bucket.find({ filename }).toArray();\n    if (files.length === 0) {\n      throw new Error('FileNotFound');\n    }\n    const parts = req\n      .get('Range')\n      .replace(/bytes=/, '')\n      .split('-');\n    const partialstart = parts[0];\n    const partialend = parts[1];\n\n    const fileLength = files[0].length;\n    const fileStart = parseInt(partialstart, 10);\n    const fileEnd = partialend ? parseInt(partialend, 10) : fileLength;\n\n    let start = Math.min(fileStart || 0, fileEnd, fileLength);\n    let end = Math.max(fileStart || 0, fileEnd) + 1 || fileLength;\n    if (isNaN(fileStart)) {\n      start = fileLength - end + 1;\n      end = fileLength;\n    }\n    end = Math.min(end, fileLength);\n    start = Math.max(start, 0);\n\n    res.status(206);\n    res.header('Accept-Ranges', 'bytes');\n    res.header('Content-Length', end - start);\n    res.header('Content-Range', 'bytes ' + start + '-' + end + '/' + fileLength);\n    res.header('Content-Type', contentType);\n    const stream = bucket.openDownloadStreamByName(filename);\n    stream.start(start);\n    if (end) {\n      stream.end(end);\n    }\n    stream.on('data', chunk => {\n      res.write(chunk);\n    });\n    stream.on('error', e => {\n      res.status(404);\n      res.send(e.message);\n    });\n    stream.on('end', () => {\n      res.end();\n    });\n  }\n\n  handleShutdown() {\n    if (!this._client) {\n      return Promise.resolve();\n    }\n    return this._client.close(false);\n  }\n\n  validateFilename(filename) {\n    return validateFilename(filename);\n  }\n}\n\nexport default GridFSBucketAdapter;\n"],"mappings":"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","ignoreList":[]}
@@ -47,6 +47,11 @@ class AdaptableController {
47
47
 
48
48
  // Makes sure the prototype matches
49
49
  const mismatches = Object.getOwnPropertyNames(Type.prototype).reduce((obj, key) => {
50
+ // Skip getters — they provide optional defaults that adapters don't need to implement
51
+ const descriptor = Object.getOwnPropertyDescriptor(Type.prototype, key);
52
+ if (descriptor && typeof descriptor.get === 'function') {
53
+ return obj;
54
+ }
50
55
  const adapterType = typeof adapter[key];
51
56
  const expectedType = typeof Type.prototype[key];
52
57
  if (adapterType !== expectedType) {
@@ -64,4 +69,4 @@ class AdaptableController {
64
69
  }
65
70
  exports.AdaptableController = AdaptableController;
66
71
  var _default = exports.default = AdaptableController;
67
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfYWRhcHRlciIsIlN5bWJvbCIsIkFkYXB0YWJsZUNvbnRyb2xsZXIiLCJjb25zdHJ1Y3RvciIsImFkYXB0ZXIiLCJhcHBJZCIsIm9wdGlvbnMiLCJ2YWxpZGF0ZUFkYXB0ZXIiLCJleHBlY3RlZEFkYXB0ZXJUeXBlIiwiRXJyb3IiLCJzZWxmIiwiRXhwZWN0ZWRUeXBlIiwibmFtZSIsIlR5cGUiLCJtaXNtYXRjaGVzIiwiT2JqZWN0IiwiZ2V0T3duUHJvcGVydHlOYW1lcyIsInByb3RvdHlwZSIsInJlZHVjZSIsIm9iaiIsImtleSIsImFkYXB0ZXJUeXBlIiwiZXhwZWN0ZWRUeXBlIiwiZXhwZWN0ZWQiLCJhY3R1YWwiLCJrZXlzIiwibGVuZ3RoIiwiZXhwb3J0cyIsIl9kZWZhdWx0IiwiZGVmYXVsdCJdLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9Db250cm9sbGVycy9BZGFwdGFibGVDb250cm9sbGVyLmpzIl0sInNvdXJjZXNDb250ZW50IjpbIi8qXG5BZGFwdGFibGVDb250cm9sbGVyLmpzXG5cbkFkYXB0YWJsZUNvbnRyb2xsZXIgaXMgdGhlIGJhc2UgY2xhc3MgZm9yIGFsbCBjb250cm9sbGVyc1xudGhhdCBzdXBwb3J0IGFkYXB0ZXIsXG5UaGUgc3VwZXIgY2xhc3MgdGFrZXMgY2FyZSBvZiBjcmVhdGluZyB0aGUgcmlnaHQgaW5zdGFuY2UgZm9yIHRoZSBhZGFwdGVyXG5iYXNlZCBvbiB0aGUgcGFyYW1ldGVycyBwYXNzZWRcblxuICovXG5cbi8vIF9hZGFwdGVyIGlzIHByaXZhdGUsIHVzZSBTeW1ib2xcbnZhciBfYWRhcHRlciA9IFN5bWJvbCgpO1xuXG5leHBvcnQgY2xhc3MgQWRhcHRhYmxlQ29udHJvbGxlciB7XG4gIGNvbnN0cnVjdG9yKGFkYXB0ZXIsIGFwcElkLCBvcHRpb25zKSB7XG4gICAgdGhpcy5vcHRpb25zID0gb3B0aW9ucztcbiAgICB0aGlzLmFwcElkID0gYXBwSWQ7XG4gICAgdGhpcy5hZGFwdGVyID0gYWRhcHRlcjtcbiAgfVxuXG4gIHNldCBhZGFwdGVyKGFkYXB0ZXIpIHtcbiAgICB0aGlzLnZhbGlkYXRlQWRhcHRlcihhZGFwdGVyKTtcbiAgICB0aGlzW19hZGFwdGVyXSA9IGFkYXB0ZXI7XG4gIH1cblxuICBnZXQgYWRhcHRlcigpIHtcbiAgICByZXR1cm4gdGhpc1tfYWRhcHRlcl07XG4gIH1cblxuICBleHBlY3RlZEFkYXB0ZXJUeXBlKCkge1xuICAgIHRocm93IG5ldyBFcnJvcignU3ViY2xhc3NlcyBzaG91bGQgaW1wbGVtZW50IGV4cGVjdGVkQWRhcHRlclR5cGUoKScpO1xuICB9XG5cbiAgdmFsaWRhdGVBZGFwdGVyKGFkYXB0ZXIpIHtcbiAgICBBZGFwdGFibGVDb250cm9sbGVyLnZhbGlkYXRlQWRhcHRlcihhZGFwdGVyLCB0aGlzKTtcbiAgfVxuXG4gIHN0YXRpYyB2YWxpZGF0ZUFkYXB0ZXIoYWRhcHRlciwgc2VsZiwgRXhwZWN0ZWRUeXBlKSB7XG4gICAgaWYgKCFhZGFwdGVyKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IodGhpcy5jb25zdHJ1Y3Rvci5uYW1lICsgJyByZXF1aXJlcyBhbiBhZGFwdGVyJyk7XG4gICAgfVxuXG4gICAgY29uc3QgVHlwZSA9IEV4cGVjdGVkVHlwZSB8fCBzZWxmLmV4cGVjdGVkQWRhcHRlclR5cGUoKTtcbiAgICAvLyBBbGxvdyBza2lwcGluZyBmb3IgdGVzdGluZ1xuICAgIGlmICghVHlwZSkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIC8vIE1ha2VzIHN1cmUgdGhlIHByb3RvdHlwZSBtYXRjaGVzXG4gICAgY29uc3QgbWlzbWF0Y2hlcyA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eU5hbWVzKFR5cGUucHJvdG90eXBlKS5yZWR1Y2UoKG9iaiwga2V5KSA9PiB7XG4gICAgICBjb25zdCBhZGFwdGVyVHlwZSA9IHR5cGVvZiBhZGFwdGVyW2tleV07XG4gICAgICBjb25zdCBleHBlY3RlZFR5cGUgPSB0eXBlb2YgVHlwZS5wcm90b3R5cGVba2V5XTtcbiAgICAgIGlmIChhZGFwdGVyVHlwZSAhPT0gZXhwZWN0ZWRUeXBlKSB7XG4gICAgICAgIG9ialtrZXldID0ge1xuICAgICAgICAgIGV4cGVjdGVkOiBleHBlY3RlZFR5cGUsXG4gICAgICAgICAgYWN0dWFsOiBhZGFwdGVyVHlwZSxcbiAgICAgICAgfTtcbiAgICAgIH1cbiAgICAgIHJldHVybiBvYmo7XG4gICAgfSwge30pO1xuXG4gICAgaWYgKE9iamVjdC5rZXlzKG1pc21hdGNoZXMpLmxlbmd0aCA+IDApIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihcIkFkYXB0ZXIgcHJvdG90eXBlIGRvbid0IG1hdGNoIGV4cGVjdGVkIHByb3RvdHlwZVwiLCBhZGFwdGVyLCBtaXNtYXRjaGVzKTtcbiAgICB9XG4gIH1cbn1cblxuZXhwb3J0IGRlZmF1bHQgQWRhcHRhYmxlQ29udHJvbGxlcjtcbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7O0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQUVBO0FBQ0EsSUFBSUEsUUFBUSxHQUFHQyxNQUFNLENBQUMsQ0FBQztBQUVoQixNQUFNQyxtQkFBbUIsQ0FBQztFQUMvQkMsV0FBV0EsQ0FBQ0MsT0FBTyxFQUFFQyxLQUFLLEVBQUVDLE9BQU8sRUFBRTtJQUNuQyxJQUFJLENBQUNBLE9BQU8sR0FBR0EsT0FBTztJQUN0QixJQUFJLENBQUNELEtBQUssR0FBR0EsS0FBSztJQUNsQixJQUFJLENBQUNELE9BQU8sR0FBR0EsT0FBTztFQUN4QjtFQUVBLElBQUlBLE9BQU9BLENBQUNBLE9BQU8sRUFBRTtJQUNuQixJQUFJLENBQUNHLGVBQWUsQ0FBQ0gsT0FBTyxDQUFDO0lBQzdCLElBQUksQ0FBQ0osUUFBUSxDQUFDLEdBQUdJLE9BQU87RUFDMUI7RUFFQSxJQUFJQSxPQUFPQSxDQUFBLEVBQUc7SUFDWixPQUFPLElBQUksQ0FBQ0osUUFBUSxDQUFDO0VBQ3ZCO0VBRUFRLG1CQUFtQkEsQ0FBQSxFQUFHO0lBQ3BCLE1BQU0sSUFBSUMsS0FBSyxDQUFDLG1EQUFtRCxDQUFDO0VBQ3RFO0VBRUFGLGVBQWVBLENBQUNILE9BQU8sRUFBRTtJQUN2QkYsbUJBQW1CLENBQUNLLGVBQWUsQ0FBQ0gsT0FBTyxFQUFFLElBQUksQ0FBQztFQUNwRDtFQUVBLE9BQU9HLGVBQWVBLENBQUNILE9BQU8sRUFBRU0sSUFBSSxFQUFFQyxZQUFZLEVBQUU7SUFDbEQsSUFBSSxDQUFDUCxPQUFPLEVBQUU7TUFDWixNQUFNLElBQUlLLEtBQUssQ0FBQyxJQUFJLENBQUNOLFdBQVcsQ0FBQ1MsSUFBSSxHQUFHLHNCQUFzQixDQUFDO0lBQ2pFO0lBRUEsTUFBTUMsSUFBSSxHQUFHRixZQUFZLElBQUlELElBQUksQ0FBQ0YsbUJBQW1CLENBQUMsQ0FBQztJQUN2RDtJQUNBLElBQUksQ0FBQ0ssSUFBSSxFQUFFO01BQ1Q7SUFDRjs7SUFFQTtJQUNBLE1BQU1DLFVBQVUsR0FBR0MsTUFBTSxDQUFDQyxtQkFBbUIsQ0FBQ0gsSUFBSSxDQUFDSSxTQUFTLENBQUMsQ0FBQ0MsTUFBTSxDQUFDLENBQUNDLEdBQUcsRUFBRUMsR0FBRyxLQUFLO01BQ2pGLE1BQU1DLFdBQVcsR0FBRyxPQUFPakIsT0FBTyxDQUFDZ0IsR0FBRyxDQUFDO01BQ3ZDLE1BQU1FLFlBQVksR0FBRyxPQUFPVCxJQUFJLENBQUNJLFNBQVMsQ0FBQ0csR0FBRyxDQUFDO01BQy9DLElBQUlDLFdBQVcsS0FBS0MsWUFBWSxFQUFFO1FBQ2hDSCxHQUFHLENBQUNDLEdBQUcsQ0FBQyxHQUFHO1VBQ1RHLFFBQVEsRUFBRUQsWUFBWTtVQUN0QkUsTUFBTSxFQUFFSDtRQUNWLENBQUM7TUFDSDtNQUNBLE9BQU9GLEdBQUc7SUFDWixDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7SUFFTixJQUFJSixNQUFNLENBQUNVLElBQUksQ0FBQ1gsVUFBVSxDQUFDLENBQUNZLE1BQU0sR0FBRyxDQUFDLEVBQUU7TUFDdEMsTUFBTSxJQUFJakIsS0FBSyxDQUFDLGtEQUFrRCxFQUFFTCxPQUFPLEVBQUVVLFVBQVUsQ0FBQztJQUMxRjtFQUNGO0FBQ0Y7QUFBQ2EsT0FBQSxDQUFBekIsbUJBQUEsR0FBQUEsbUJBQUE7QUFBQSxJQUFBMEIsUUFBQSxHQUFBRCxPQUFBLENBQUFFLE9BQUEsR0FFYzNCLG1CQUFtQiIsImlnbm9yZUxpc3QiOltdfQ==
72
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfYWRhcHRlciIsIlN5bWJvbCIsIkFkYXB0YWJsZUNvbnRyb2xsZXIiLCJjb25zdHJ1Y3RvciIsImFkYXB0ZXIiLCJhcHBJZCIsIm9wdGlvbnMiLCJ2YWxpZGF0ZUFkYXB0ZXIiLCJleHBlY3RlZEFkYXB0ZXJUeXBlIiwiRXJyb3IiLCJzZWxmIiwiRXhwZWN0ZWRUeXBlIiwibmFtZSIsIlR5cGUiLCJtaXNtYXRjaGVzIiwiT2JqZWN0IiwiZ2V0T3duUHJvcGVydHlOYW1lcyIsInByb3RvdHlwZSIsInJlZHVjZSIsIm9iaiIsImtleSIsImRlc2NyaXB0b3IiLCJnZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3IiLCJnZXQiLCJhZGFwdGVyVHlwZSIsImV4cGVjdGVkVHlwZSIsImV4cGVjdGVkIiwiYWN0dWFsIiwia2V5cyIsImxlbmd0aCIsImV4cG9ydHMiLCJfZGVmYXVsdCIsImRlZmF1bHQiXSwic291cmNlcyI6WyIuLi8uLi9zcmMvQ29udHJvbGxlcnMvQWRhcHRhYmxlQ29udHJvbGxlci5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuQWRhcHRhYmxlQ29udHJvbGxlci5qc1xuXG5BZGFwdGFibGVDb250cm9sbGVyIGlzIHRoZSBiYXNlIGNsYXNzIGZvciBhbGwgY29udHJvbGxlcnNcbnRoYXQgc3VwcG9ydCBhZGFwdGVyLFxuVGhlIHN1cGVyIGNsYXNzIHRha2VzIGNhcmUgb2YgY3JlYXRpbmcgdGhlIHJpZ2h0IGluc3RhbmNlIGZvciB0aGUgYWRhcHRlclxuYmFzZWQgb24gdGhlIHBhcmFtZXRlcnMgcGFzc2VkXG5cbiAqL1xuXG4vLyBfYWRhcHRlciBpcyBwcml2YXRlLCB1c2UgU3ltYm9sXG52YXIgX2FkYXB0ZXIgPSBTeW1ib2woKTtcblxuZXhwb3J0IGNsYXNzIEFkYXB0YWJsZUNvbnRyb2xsZXIge1xuICBjb25zdHJ1Y3RvcihhZGFwdGVyLCBhcHBJZCwgb3B0aW9ucykge1xuICAgIHRoaXMub3B0aW9ucyA9IG9wdGlvbnM7XG4gICAgdGhpcy5hcHBJZCA9IGFwcElkO1xuICAgIHRoaXMuYWRhcHRlciA9IGFkYXB0ZXI7XG4gIH1cblxuICBzZXQgYWRhcHRlcihhZGFwdGVyKSB7XG4gICAgdGhpcy52YWxpZGF0ZUFkYXB0ZXIoYWRhcHRlcik7XG4gICAgdGhpc1tfYWRhcHRlcl0gPSBhZGFwdGVyO1xuICB9XG5cbiAgZ2V0IGFkYXB0ZXIoKSB7XG4gICAgcmV0dXJuIHRoaXNbX2FkYXB0ZXJdO1xuICB9XG5cbiAgZXhwZWN0ZWRBZGFwdGVyVHlwZSgpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ1N1YmNsYXNzZXMgc2hvdWxkIGltcGxlbWVudCBleHBlY3RlZEFkYXB0ZXJUeXBlKCknKTtcbiAgfVxuXG4gIHZhbGlkYXRlQWRhcHRlcihhZGFwdGVyKSB7XG4gICAgQWRhcHRhYmxlQ29udHJvbGxlci52YWxpZGF0ZUFkYXB0ZXIoYWRhcHRlciwgdGhpcyk7XG4gIH1cblxuICBzdGF0aWMgdmFsaWRhdGVBZGFwdGVyKGFkYXB0ZXIsIHNlbGYsIEV4cGVjdGVkVHlwZSkge1xuICAgIGlmICghYWRhcHRlcikge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKHRoaXMuY29uc3RydWN0b3IubmFtZSArICcgcmVxdWlyZXMgYW4gYWRhcHRlcicpO1xuICAgIH1cblxuICAgIGNvbnN0IFR5cGUgPSBFeHBlY3RlZFR5cGUgfHwgc2VsZi5leHBlY3RlZEFkYXB0ZXJUeXBlKCk7XG4gICAgLy8gQWxsb3cgc2tpcHBpbmcgZm9yIHRlc3RpbmdcbiAgICBpZiAoIVR5cGUpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICAvLyBNYWtlcyBzdXJlIHRoZSBwcm90b3R5cGUgbWF0Y2hlc1xuICAgIGNvbnN0IG1pc21hdGNoZXMgPSBPYmplY3QuZ2V0T3duUHJvcGVydHlOYW1lcyhUeXBlLnByb3RvdHlwZSkucmVkdWNlKChvYmosIGtleSkgPT4ge1xuICAgICAgLy8gU2tpcCBnZXR0ZXJzIOKAlCB0aGV5IHByb3ZpZGUgb3B0aW9uYWwgZGVmYXVsdHMgdGhhdCBhZGFwdGVycyBkb24ndCBuZWVkIHRvIGltcGxlbWVudFxuICAgICAgY29uc3QgZGVzY3JpcHRvciA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3IoVHlwZS5wcm90b3R5cGUsIGtleSk7XG4gICAgICBpZiAoZGVzY3JpcHRvciAmJiB0eXBlb2YgZGVzY3JpcHRvci5nZXQgPT09ICdmdW5jdGlvbicpIHtcbiAgICAgICAgcmV0dXJuIG9iajtcbiAgICAgIH1cbiAgICAgIGNvbnN0IGFkYXB0ZXJUeXBlID0gdHlwZW9mIGFkYXB0ZXJba2V5XTtcbiAgICAgIGNvbnN0IGV4cGVjdGVkVHlwZSA9IHR5cGVvZiBUeXBlLnByb3RvdHlwZVtrZXldO1xuICAgICAgaWYgKGFkYXB0ZXJUeXBlICE9PSBleHBlY3RlZFR5cGUpIHtcbiAgICAgICAgb2JqW2tleV0gPSB7XG4gICAgICAgICAgZXhwZWN0ZWQ6IGV4cGVjdGVkVHlwZSxcbiAgICAgICAgICBhY3R1YWw6IGFkYXB0ZXJUeXBlLFxuICAgICAgICB9O1xuICAgICAgfVxuICAgICAgcmV0dXJuIG9iajtcbiAgICB9LCB7fSk7XG5cbiAgICBpZiAoT2JqZWN0LmtleXMobWlzbWF0Y2hlcykubGVuZ3RoID4gMCkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFwiQWRhcHRlciBwcm90b3R5cGUgZG9uJ3QgbWF0Y2ggZXhwZWN0ZWQgcHJvdG90eXBlXCIsIGFkYXB0ZXIsIG1pc21hdGNoZXMpO1xuICAgIH1cbiAgfVxufVxuXG5leHBvcnQgZGVmYXVsdCBBZGFwdGFibGVDb250cm9sbGVyO1xuIl0sIm1hcHBpbmdzIjoiOzs7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQSxJQUFJQSxRQUFRLEdBQUdDLE1BQU0sQ0FBQyxDQUFDO0FBRWhCLE1BQU1DLG1CQUFtQixDQUFDO0VBQy9CQyxXQUFXQSxDQUFDQyxPQUFPLEVBQUVDLEtBQUssRUFBRUMsT0FBTyxFQUFFO0lBQ25DLElBQUksQ0FBQ0EsT0FBTyxHQUFHQSxPQUFPO0lBQ3RCLElBQUksQ0FBQ0QsS0FBSyxHQUFHQSxLQUFLO0lBQ2xCLElBQUksQ0FBQ0QsT0FBTyxHQUFHQSxPQUFPO0VBQ3hCO0VBRUEsSUFBSUEsT0FBT0EsQ0FBQ0EsT0FBTyxFQUFFO0lBQ25CLElBQUksQ0FBQ0csZUFBZSxDQUFDSCxPQUFPLENBQUM7SUFDN0IsSUFBSSxDQUFDSixRQUFRLENBQUMsR0FBR0ksT0FBTztFQUMxQjtFQUVBLElBQUlBLE9BQU9BLENBQUEsRUFBRztJQUNaLE9BQU8sSUFBSSxDQUFDSixRQUFRLENBQUM7RUFDdkI7RUFFQVEsbUJBQW1CQSxDQUFBLEVBQUc7SUFDcEIsTUFBTSxJQUFJQyxLQUFLLENBQUMsbURBQW1ELENBQUM7RUFDdEU7RUFFQUYsZUFBZUEsQ0FBQ0gsT0FBTyxFQUFFO0lBQ3ZCRixtQkFBbUIsQ0FBQ0ssZUFBZSxDQUFDSCxPQUFPLEVBQUUsSUFBSSxDQUFDO0VBQ3BEO0VBRUEsT0FBT0csZUFBZUEsQ0FBQ0gsT0FBTyxFQUFFTSxJQUFJLEVBQUVDLFlBQVksRUFBRTtJQUNsRCxJQUFJLENBQUNQLE9BQU8sRUFBRTtNQUNaLE1BQU0sSUFBSUssS0FBSyxDQUFDLElBQUksQ0FBQ04sV0FBVyxDQUFDUyxJQUFJLEdBQUcsc0JBQXNCLENBQUM7SUFDakU7SUFFQSxNQUFNQyxJQUFJLEdBQUdGLFlBQVksSUFBSUQsSUFBSSxDQUFDRixtQkFBbUIsQ0FBQyxDQUFDO0lBQ3ZEO0lBQ0EsSUFBSSxDQUFDSyxJQUFJLEVBQUU7TUFDVDtJQUNGOztJQUVBO0lBQ0EsTUFBTUMsVUFBVSxHQUFHQyxNQUFNLENBQUNDLG1CQUFtQixDQUFDSCxJQUFJLENBQUNJLFNBQVMsQ0FBQyxDQUFDQyxNQUFNLENBQUMsQ0FBQ0MsR0FBRyxFQUFFQyxHQUFHLEtBQUs7TUFDakY7TUFDQSxNQUFNQyxVQUFVLEdBQUdOLE1BQU0sQ0FBQ08sd0JBQXdCLENBQUNULElBQUksQ0FBQ0ksU0FBUyxFQUFFRyxHQUFHLENBQUM7TUFDdkUsSUFBSUMsVUFBVSxJQUFJLE9BQU9BLFVBQVUsQ0FBQ0UsR0FBRyxLQUFLLFVBQVUsRUFBRTtRQUN0RCxPQUFPSixHQUFHO01BQ1o7TUFDQSxNQUFNSyxXQUFXLEdBQUcsT0FBT3BCLE9BQU8sQ0FBQ2dCLEdBQUcsQ0FBQztNQUN2QyxNQUFNSyxZQUFZLEdBQUcsT0FBT1osSUFBSSxDQUFDSSxTQUFTLENBQUNHLEdBQUcsQ0FBQztNQUMvQyxJQUFJSSxXQUFXLEtBQUtDLFlBQVksRUFBRTtRQUNoQ04sR0FBRyxDQUFDQyxHQUFHLENBQUMsR0FBRztVQUNUTSxRQUFRLEVBQUVELFlBQVk7VUFDdEJFLE1BQU0sRUFBRUg7UUFDVixDQUFDO01BQ0g7TUFDQSxPQUFPTCxHQUFHO0lBQ1osQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO0lBRU4sSUFBSUosTUFBTSxDQUFDYSxJQUFJLENBQUNkLFVBQVUsQ0FBQyxDQUFDZSxNQUFNLEdBQUcsQ0FBQyxFQUFFO01BQ3RDLE1BQU0sSUFBSXBCLEtBQUssQ0FBQyxrREFBa0QsRUFBRUwsT0FBTyxFQUFFVSxVQUFVLENBQUM7SUFDMUY7RUFDRjtBQUNGO0FBQUNnQixPQUFBLENBQUE1QixtQkFBQSxHQUFBQSxtQkFBQTtBQUFBLElBQUE2QixRQUFBLEdBQUFELE9BQUEsQ0FBQUUsT0FBQSxHQUVjOUIsbUJBQW1CIiwiaWdub3JlTGlzdCI6W119
@@ -29,6 +29,16 @@ class FilesController extends _AdaptableController.default {
29
29
  if (!this.options.preserveFileName) {
30
30
  filename = (0, _cryptoUtils.randomHexString)(32) + '_' + filename;
31
31
  }
32
+
33
+ // Fallback: buffer stream for adapters that don't support streaming
34
+ if (typeof data?.pipe === 'function' && !this.adapter.supportsStreaming) {
35
+ data = await new Promise((resolve, reject) => {
36
+ const chunks = [];
37
+ data.on('data', chunk => chunks.push(chunk));
38
+ data.on('end', () => resolve(Buffer.concat(chunks)));
39
+ data.on('error', reject);
40
+ });
41
+ }
32
42
  const location = await this.adapter.getFileLocation(config, filename);
33
43
  await this.adapter.createFile(filename, data, contentType, options);
34
44
  return {
@@ -103,4 +113,4 @@ class FilesController extends _AdaptableController.default {
103
113
  }
104
114
  exports.FilesController = FilesController;
105
115
  var _default = exports.default = FilesController;
106
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_cryptoUtils","require","_AdaptableController","_interopRequireDefault","_FilesAdapter","_path","e","__esModule","default","Parse","legacyFilesRegex","RegExp","FilesController","AdaptableController","getFileData","config","filename","adapter","createFile","data","contentType","options","extname","path","hasExtension","length","mime","getExtension","getType","preserveFileName","randomHexString","location","getFileLocation","url","name","deleteFile","getMetadata","Promise","resolve","expandFilesInObject","object","Array","promises","map","obj","all","key","fileObject","fileKey","undefined","indexOf","encodeURIComponent","test","expectedAdapterType","FilesAdapter","handleFileStream","req","res","validateFilename","error","Error","INVALID_FILE_NAME","exports","_default"],"sources":["../../src/Controllers/FilesController.js"],"sourcesContent":["// FilesController.js\nimport { randomHexString } from '../cryptoUtils';\nimport AdaptableController from './AdaptableController';\nimport { validateFilename, FilesAdapter } from '../Adapters/Files/FilesAdapter';\nimport path from 'path';\nconst Parse = require('parse/node').Parse;\n\nconst legacyFilesRegex = new RegExp(\n  '^[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}-.*'\n);\n\nexport class FilesController extends AdaptableController {\n  getFileData(config, filename) {\n    return this.adapter.getFileData(filename);\n  }\n\n  async createFile(config, filename, data, contentType, options) {\n    const extname = path.extname(filename);\n\n    const hasExtension = extname.length > 0;\n    const mime = (await import('mime')).default\n    if (!hasExtension && contentType && mime.getExtension(contentType)) {\n      filename = filename + '.' + mime.getExtension(contentType);\n    } else if (hasExtension && !contentType) {\n      contentType = mime.getType(filename);\n    }\n\n    if (!this.options.preserveFileName) {\n      filename = randomHexString(32) + '_' + filename;\n    }\n\n    const location = await this.adapter.getFileLocation(config, filename);\n    await this.adapter.createFile(filename, data, contentType, options);\n    return {\n      url: location,\n      name: filename,\n    }\n  }\n\n  deleteFile(config, filename) {\n    return this.adapter.deleteFile(filename);\n  }\n\n  getMetadata(filename) {\n    if (typeof this.adapter.getMetadata === 'function') {\n      return this.adapter.getMetadata(filename);\n    }\n    return Promise.resolve({});\n  }\n\n  /**\n   * Find file references in REST-format object and adds the url key\n   * with the current mount point and app id.\n   * Object may be a single object or list of REST-format objects.\n   */\n  async expandFilesInObject(config, object) {\n    if (object instanceof Array) {\n      const promises = object.map(obj => this.expandFilesInObject(config, obj));\n      await Promise.all(promises);\n      return;\n    }\n    if (typeof object !== 'object') {\n      return;\n    }\n    for (const key in object) {\n      const fileObject = object[key];\n      if (fileObject && fileObject['__type'] === 'File') {\n        if (fileObject['url']) {\n          continue;\n        }\n        const filename = fileObject['name'];\n        // all filenames starting with \"tfss-\" should be from files.parsetfss.com\n        // all filenames starting with a \"-\" seperated UUID should be from files.parse.com\n        // all other filenames have been migrated or created from Parse Server\n        if (config.fileKey === undefined) {\n          fileObject['url'] = await this.adapter.getFileLocation(config, filename);\n        } else {\n          if (filename.indexOf('tfss-') === 0) {\n            fileObject['url'] =\n              'http://files.parsetfss.com/' + config.fileKey + '/' + encodeURIComponent(filename);\n          } else if (legacyFilesRegex.test(filename)) {\n            fileObject['url'] =\n              'http://files.parse.com/' + config.fileKey + '/' + encodeURIComponent(filename);\n          } else {\n            fileObject['url'] = await this.adapter.getFileLocation(config, filename);\n          }\n        }\n      }\n    }\n  }\n\n  expectedAdapterType() {\n    return FilesAdapter;\n  }\n\n  handleFileStream(config, filename, req, res, contentType) {\n    return this.adapter.handleFileStream(filename, req, res, contentType);\n  }\n\n  validateFilename(filename) {\n    if (typeof this.adapter.validateFilename === 'function') {\n      const error = this.adapter.validateFilename(filename);\n      if (typeof error !== 'string') {\n        return error;\n      }\n      return new Parse.Error(Parse.Error.INVALID_FILE_NAME, error);\n    }\n    return validateFilename(filename);\n  }\n}\n\nexport default FilesController;\n"],"mappings":"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","ignoreList":[]}
116
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_cryptoUtils","require","_AdaptableController","_interopRequireDefault","_FilesAdapter","_path","e","__esModule","default","Parse","legacyFilesRegex","RegExp","FilesController","AdaptableController","getFileData","config","filename","adapter","createFile","data","contentType","options","extname","path","hasExtension","length","mime","getExtension","getType","preserveFileName","randomHexString","pipe","supportsStreaming","Promise","resolve","reject","chunks","on","chunk","push","Buffer","concat","location","getFileLocation","url","name","deleteFile","getMetadata","expandFilesInObject","object","Array","promises","map","obj","all","key","fileObject","fileKey","undefined","indexOf","encodeURIComponent","test","expectedAdapterType","FilesAdapter","handleFileStream","req","res","validateFilename","error","Error","INVALID_FILE_NAME","exports","_default"],"sources":["../../src/Controllers/FilesController.js"],"sourcesContent":["// FilesController.js\nimport { randomHexString } from '../cryptoUtils';\nimport AdaptableController from './AdaptableController';\nimport { validateFilename, FilesAdapter } from '../Adapters/Files/FilesAdapter';\nimport path from 'path';\nconst Parse = require('parse/node').Parse;\n\nconst legacyFilesRegex = new RegExp(\n  '^[0-9a-fA-F]{8}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{4}-[0-9a-fA-F]{12}-.*'\n);\n\nexport class FilesController extends AdaptableController {\n  getFileData(config, filename) {\n    return this.adapter.getFileData(filename);\n  }\n\n  async createFile(config, filename, data, contentType, options) {\n    const extname = path.extname(filename);\n\n    const hasExtension = extname.length > 0;\n    const mime = (await import('mime')).default\n    if (!hasExtension && contentType && mime.getExtension(contentType)) {\n      filename = filename + '.' + mime.getExtension(contentType);\n    } else if (hasExtension && !contentType) {\n      contentType = mime.getType(filename);\n    }\n\n    if (!this.options.preserveFileName) {\n      filename = randomHexString(32) + '_' + filename;\n    }\n\n    // Fallback: buffer stream for adapters that don't support streaming\n    if (typeof data?.pipe === 'function' && !this.adapter.supportsStreaming) {\n      data = await new Promise((resolve, reject) => {\n        const chunks = [];\n        data.on('data', chunk => chunks.push(chunk));\n        data.on('end', () => resolve(Buffer.concat(chunks)));\n        data.on('error', reject);\n      });\n    }\n\n    const location = await this.adapter.getFileLocation(config, filename);\n    await this.adapter.createFile(filename, data, contentType, options);\n    return {\n      url: location,\n      name: filename,\n    }\n  }\n\n  deleteFile(config, filename) {\n    return this.adapter.deleteFile(filename);\n  }\n\n  getMetadata(filename) {\n    if (typeof this.adapter.getMetadata === 'function') {\n      return this.adapter.getMetadata(filename);\n    }\n    return Promise.resolve({});\n  }\n\n  /**\n   * Find file references in REST-format object and adds the url key\n   * with the current mount point and app id.\n   * Object may be a single object or list of REST-format objects.\n   */\n  async expandFilesInObject(config, object) {\n    if (object instanceof Array) {\n      const promises = object.map(obj => this.expandFilesInObject(config, obj));\n      await Promise.all(promises);\n      return;\n    }\n    if (typeof object !== 'object') {\n      return;\n    }\n    for (const key in object) {\n      const fileObject = object[key];\n      if (fileObject && fileObject['__type'] === 'File') {\n        if (fileObject['url']) {\n          continue;\n        }\n        const filename = fileObject['name'];\n        // all filenames starting with \"tfss-\" should be from files.parsetfss.com\n        // all filenames starting with a \"-\" seperated UUID should be from files.parse.com\n        // all other filenames have been migrated or created from Parse Server\n        if (config.fileKey === undefined) {\n          fileObject['url'] = await this.adapter.getFileLocation(config, filename);\n        } else {\n          if (filename.indexOf('tfss-') === 0) {\n            fileObject['url'] =\n              'http://files.parsetfss.com/' + config.fileKey + '/' + encodeURIComponent(filename);\n          } else if (legacyFilesRegex.test(filename)) {\n            fileObject['url'] =\n              'http://files.parse.com/' + config.fileKey + '/' + encodeURIComponent(filename);\n          } else {\n            fileObject['url'] = await this.adapter.getFileLocation(config, filename);\n          }\n        }\n      }\n    }\n  }\n\n  expectedAdapterType() {\n    return FilesAdapter;\n  }\n\n  handleFileStream(config, filename, req, res, contentType) {\n    return this.adapter.handleFileStream(filename, req, res, contentType);\n  }\n\n  validateFilename(filename) {\n    if (typeof this.adapter.validateFilename === 'function') {\n      const error = this.adapter.validateFilename(filename);\n      if (typeof error !== 'string') {\n        return error;\n      }\n      return new Parse.Error(Parse.Error.INVALID_FILE_NAME, error);\n    }\n    return validateFilename(filename);\n  }\n}\n\nexport default FilesController;\n"],"mappings":";;;;;;AACA,IAAAA,YAAA,GAAAC,OAAA;AACA,IAAAC,oBAAA,GAAAC,sBAAA,CAAAF,OAAA;AACA,IAAAG,aAAA,GAAAH,OAAA;AACA,IAAAI,KAAA,GAAAF,sBAAA,CAAAF,OAAA;AAAwB,SAAAE,uBAAAG,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAC,UAAA,GAAAD,CAAA,KAAAE,OAAA,EAAAF,CAAA;AAJxB;;AAKA,MAAMG,KAAK,GAAGR,OAAO,CAAC,YAAY,CAAC,CAACQ,KAAK;AAEzC,MAAMC,gBAAgB,GAAG,IAAIC,MAAM,CACjC,iFACF,CAAC;AAEM,MAAMC,eAAe,SAASC,4BAAmB,CAAC;EACvDC,WAAWA,CAACC,MAAM,EAAEC,QAAQ,EAAE;IAC5B,OAAO,IAAI,CAACC,OAAO,CAACH,WAAW,CAACE,QAAQ,CAAC;EAC3C;EAEA,MAAME,UAAUA,CAACH,MAAM,EAAEC,QAAQ,EAAEG,IAAI,EAAEC,WAAW,EAAEC,OAAO,EAAE;IAC7D,MAAMC,OAAO,GAAGC,aAAI,CAACD,OAAO,CAACN,QAAQ,CAAC;IAEtC,MAAMQ,YAAY,GAAGF,OAAO,CAACG,MAAM,GAAG,CAAC;IACvC,MAAMC,IAAI,GAAG,CAAC,MAAM,MAAM,CAAC,MAAM,CAAC,EAAElB,OAAO;IAC3C,IAAI,CAACgB,YAAY,IAAIJ,WAAW,IAAIM,IAAI,CAACC,YAAY,CAACP,WAAW,CAAC,EAAE;MAClEJ,QAAQ,GAAGA,QAAQ,GAAG,GAAG,GAAGU,IAAI,CAACC,YAAY,CAACP,WAAW,CAAC;IAC5D,CAAC,MAAM,IAAII,YAAY,IAAI,CAACJ,WAAW,EAAE;MACvCA,WAAW,GAAGM,IAAI,CAACE,OAAO,CAACZ,QAAQ,CAAC;IACtC;IAEA,IAAI,CAAC,IAAI,CAACK,OAAO,CAACQ,gBAAgB,EAAE;MAClCb,QAAQ,GAAG,IAAAc,4BAAe,EAAC,EAAE,CAAC,GAAG,GAAG,GAAGd,QAAQ;IACjD;;IAEA;IACA,IAAI,OAAOG,IAAI,EAAEY,IAAI,KAAK,UAAU,IAAI,CAAC,IAAI,CAACd,OAAO,CAACe,iBAAiB,EAAE;MACvEb,IAAI,GAAG,MAAM,IAAIc,OAAO,CAAC,CAACC,OAAO,EAAEC,MAAM,KAAK;QAC5C,MAAMC,MAAM,GAAG,EAAE;QACjBjB,IAAI,CAACkB,EAAE,CAAC,MAAM,EAAEC,KAAK,IAAIF,MAAM,CAACG,IAAI,CAACD,KAAK,CAAC,CAAC;QAC5CnB,IAAI,CAACkB,EAAE,CAAC,KAAK,EAAE,MAAMH,OAAO,CAACM,MAAM,CAACC,MAAM,CAACL,MAAM,CAAC,CAAC,CAAC;QACpDjB,IAAI,CAACkB,EAAE,CAAC,OAAO,EAAEF,MAAM,CAAC;MAC1B,CAAC,CAAC;IACJ;IAEA,MAAMO,QAAQ,GAAG,MAAM,IAAI,CAACzB,OAAO,CAAC0B,eAAe,CAAC5B,MAAM,EAAEC,QAAQ,CAAC;IACrE,MAAM,IAAI,CAACC,OAAO,CAACC,UAAU,CAACF,QAAQ,EAAEG,IAAI,EAAEC,WAAW,EAAEC,OAAO,CAAC;IACnE,OAAO;MACLuB,GAAG,EAAEF,QAAQ;MACbG,IAAI,EAAE7B;IACR,CAAC;EACH;EAEA8B,UAAUA,CAAC/B,MAAM,EAAEC,QAAQ,EAAE;IAC3B,OAAO,IAAI,CAACC,OAAO,CAAC6B,UAAU,CAAC9B,QAAQ,CAAC;EAC1C;EAEA+B,WAAWA,CAAC/B,QAAQ,EAAE;IACpB,IAAI,OAAO,IAAI,CAACC,OAAO,CAAC8B,WAAW,KAAK,UAAU,EAAE;MAClD,OAAO,IAAI,CAAC9B,OAAO,CAAC8B,WAAW,CAAC/B,QAAQ,CAAC;IAC3C;IACA,OAAOiB,OAAO,CAACC,OAAO,CAAC,CAAC,CAAC,CAAC;EAC5B;;EAEA;AACF;AACA;AACA;AACA;EACE,MAAMc,mBAAmBA,CAACjC,MAAM,EAAEkC,MAAM,EAAE;IACxC,IAAIA,MAAM,YAAYC,KAAK,EAAE;MAC3B,MAAMC,QAAQ,GAAGF,MAAM,CAACG,GAAG,CAACC,GAAG,IAAI,IAAI,CAACL,mBAAmB,CAACjC,MAAM,EAAEsC,GAAG,CAAC,CAAC;MACzE,MAAMpB,OAAO,CAACqB,GAAG,CAACH,QAAQ,CAAC;MAC3B;IACF;IACA,IAAI,OAAOF,MAAM,KAAK,QAAQ,EAAE;MAC9B;IACF;IACA,KAAK,MAAMM,GAAG,IAAIN,MAAM,EAAE;MACxB,MAAMO,UAAU,GAAGP,MAAM,CAACM,GAAG,CAAC;MAC9B,IAAIC,UAAU,IAAIA,UAAU,CAAC,QAAQ,CAAC,KAAK,MAAM,EAAE;QACjD,IAAIA,UAAU,CAAC,KAAK,CAAC,EAAE;UACrB;QACF;QACA,MAAMxC,QAAQ,GAAGwC,UAAU,CAAC,MAAM,CAAC;QACnC;QACA;QACA;QACA,IAAIzC,MAAM,CAAC0C,OAAO,KAAKC,SAAS,EAAE;UAChCF,UAAU,CAAC,KAAK,CAAC,GAAG,MAAM,IAAI,CAACvC,OAAO,CAAC0B,eAAe,CAAC5B,MAAM,EAAEC,QAAQ,CAAC;QAC1E,CAAC,MAAM;UACL,IAAIA,QAAQ,CAAC2C,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;YACnCH,UAAU,CAAC,KAAK,CAAC,GACf,6BAA6B,GAAGzC,MAAM,CAAC0C,OAAO,GAAG,GAAG,GAAGG,kBAAkB,CAAC5C,QAAQ,CAAC;UACvF,CAAC,MAAM,IAAIN,gBAAgB,CAACmD,IAAI,CAAC7C,QAAQ,CAAC,EAAE;YAC1CwC,UAAU,CAAC,KAAK,CAAC,GACf,yBAAyB,GAAGzC,MAAM,CAAC0C,OAAO,GAAG,GAAG,GAAGG,kBAAkB,CAAC5C,QAAQ,CAAC;UACnF,CAAC,MAAM;YACLwC,UAAU,CAAC,KAAK,CAAC,GAAG,MAAM,IAAI,CAACvC,OAAO,CAAC0B,eAAe,CAAC5B,MAAM,EAAEC,QAAQ,CAAC;UAC1E;QACF;MACF;IACF;EACF;EAEA8C,mBAAmBA,CAAA,EAAG;IACpB,OAAOC,0BAAY;EACrB;EAEAC,gBAAgBA,CAACjD,MAAM,EAAEC,QAAQ,EAAEiD,GAAG,EAAEC,GAAG,EAAE9C,WAAW,EAAE;IACxD,OAAO,IAAI,CAACH,OAAO,CAAC+C,gBAAgB,CAAChD,QAAQ,EAAEiD,GAAG,EAAEC,GAAG,EAAE9C,WAAW,CAAC;EACvE;EAEA+C,gBAAgBA,CAACnD,QAAQ,EAAE;IACzB,IAAI,OAAO,IAAI,CAACC,OAAO,CAACkD,gBAAgB,KAAK,UAAU,EAAE;MACvD,MAAMC,KAAK,GAAG,IAAI,CAACnD,OAAO,CAACkD,gBAAgB,CAACnD,QAAQ,CAAC;MACrD,IAAI,OAAOoD,KAAK,KAAK,QAAQ,EAAE;QAC7B,OAAOA,KAAK;MACd;MACA,OAAO,IAAI3D,KAAK,CAAC4D,KAAK,CAAC5D,KAAK,CAAC4D,KAAK,CAACC,iBAAiB,EAAEF,KAAK,CAAC;IAC9D;IACA,OAAO,IAAAD,8BAAgB,EAACnD,QAAQ,CAAC;EACnC;AACF;AAACuD,OAAA,CAAA3D,eAAA,GAAAA,eAAA;AAAA,IAAA4D,QAAA,GAAAD,OAAA,CAAA/D,OAAA,GAEcI,eAAe","ignoreList":[]}
@@ -4,15 +4,77 @@ Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
6
  exports.FilesRouter = void 0;
7
+ exports.createSizeLimitedStream = createSizeLimitedStream;
7
8
  var _express = _interopRequireDefault(require("express"));
8
9
  var Middlewares = _interopRequireWildcard(require("../middlewares"));
9
10
  var _node = _interopRequireDefault(require("parse/node"));
10
11
  var _Config = _interopRequireDefault(require("../Config"));
11
12
  var _logger = _interopRequireDefault(require("../logger"));
13
+ var _stream = require("stream");
12
14
  function _interopRequireWildcard(e, t) { if ("function" == typeof WeakMap) var r = new WeakMap(), n = new WeakMap(); return (_interopRequireWildcard = function (e, t) { if (!t && e && e.__esModule) return e; var o, i, f = { __proto__: null, default: e }; if (null === e || "object" != typeof e && "function" != typeof e) return f; if (o = t ? n : r) { if (o.has(e)) return o.get(e); o.set(e, f); } for (const t in e) "default" !== t && {}.hasOwnProperty.call(e, t) && ((i = (o = Object.defineProperty) && Object.getOwnPropertyDescriptor(e, t)) && (i.get || i.set) ? o(f, t, i) : f[t] = e[t]); return f; })(e, t); }
13
15
  function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
14
16
  const triggers = require('../triggers');
15
17
  const Utils = require('../Utils');
18
+ /**
19
+ * Wraps a readable stream in a Readable that enforces a byte size limit.
20
+ * Data flow is lazy: the source is not read until a consumer starts reading
21
+ * from the returned stream (via pipe or 'data' listener). This ensures the
22
+ * consumer's error listener is attached before any data (or error) is emitted.
23
+ */
24
+ function createSizeLimitedStream(source, maxBytes) {
25
+ let totalBytes = 0;
26
+ let started = false;
27
+ let sourceEnded = false;
28
+ let onData, onEnd, onError;
29
+ const output = new _stream.Readable({
30
+ read() {
31
+ if (!started) {
32
+ started = true;
33
+ onData = chunk => {
34
+ totalBytes += chunk.length;
35
+ if (totalBytes > maxBytes) {
36
+ output.destroy(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, `File size exceeds maximum allowed: ${maxBytes} bytes.`));
37
+ return;
38
+ }
39
+ if (!output.push(chunk)) {
40
+ source.pause();
41
+ }
42
+ };
43
+ onEnd = () => {
44
+ sourceEnded = true;
45
+ output.push(null);
46
+ };
47
+ onError = err => output.destroy(err);
48
+ source.on('data', onData);
49
+ source.on('end', onEnd);
50
+ source.on('error', onError);
51
+ }
52
+
53
+ // Resume source in case it was paused due to backpressure
54
+ if (!sourceEnded) {
55
+ source.resume();
56
+ }
57
+ },
58
+ destroy(err, callback) {
59
+ if (onData) {
60
+ source.removeListener('data', onData);
61
+ }
62
+ if (onEnd) {
63
+ source.removeListener('end', onEnd);
64
+ }
65
+ if (onError) {
66
+ source.removeListener('error', onError);
67
+ }
68
+ // Suppress errors emitted during drain (e.g. client disconnect)
69
+ source.on('error', () => {});
70
+ if (!sourceEnded) {
71
+ source.resume();
72
+ }
73
+ callback(err);
74
+ }
75
+ });
76
+ return output;
77
+ }
16
78
  class FilesRouter {
17
79
  expressRouter({
18
80
  maxUploadSize = '20Mb'
@@ -23,14 +85,7 @@ class FilesRouter {
23
85
  router.post('/files', function (req, res, next) {
24
86
  next(new _node.default.Error(_node.default.Error.INVALID_FILE_NAME, 'Filename not provided.'));
25
87
  });
26
- router.post('/files/:filename', _express.default.raw({
27
- type: () => {
28
- return true;
29
- },
30
- limit: maxUploadSize
31
- }),
32
- // Allow uploads without Content-Type, or with any Content-Type.
33
- Middlewares.handleParseHeaders, Middlewares.handleParseSession, this.createHandler);
88
+ router.post('/files/:filename', this._bodyParsingMiddleware(maxUploadSize), Middlewares.handleParseHeaders, Middlewares.handleParseSession, this.createHandler.bind(this));
34
89
  router.delete('/files/:filename', Middlewares.handleParseHeaders, Middlewares.handleParseSession, Middlewares.enforceMasterKeyAccess, this.deleteHandler);
35
90
  return router;
36
91
  }
@@ -105,6 +160,19 @@ class FilesRouter {
105
160
  });
106
161
  }
107
162
  }
163
+ _bodyParsingMiddleware(maxUploadSize) {
164
+ const rawParser = _express.default.raw({
165
+ type: () => true,
166
+ limit: maxUploadSize
167
+ });
168
+ return (req, res, next) => {
169
+ if (req.get('X-Parse-Upload-Mode') === 'stream') {
170
+ req._maxUploadSizeBytes = Utils.parseSizeToBytes(maxUploadSize);
171
+ return next();
172
+ }
173
+ return rawParser(req, res, next);
174
+ };
175
+ }
108
176
  async createHandler(req, res, next) {
109
177
  const config = req.config;
110
178
  const user = req.auth.user;
@@ -127,10 +195,6 @@ class FilesRouter {
127
195
  filename
128
196
  } = req.params;
129
197
  const contentType = req.get('Content-type');
130
- if (!req.body || !req.body.length) {
131
- next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, 'Invalid file upload.'));
132
- return;
133
- }
134
198
  const error = filesController.validateFilename(filename);
135
199
  if (error) {
136
200
  next(error);
@@ -161,6 +225,24 @@ class FilesRouter {
161
225
  return;
162
226
  }
163
227
  }
228
+
229
+ // Dispatch to the appropriate handler based on whether the body was buffered
230
+ if (req.body instanceof Buffer) {
231
+ return this._handleBufferedUpload(req, res, next);
232
+ }
233
+ return this._handleStreamUpload(req, res, next);
234
+ }
235
+ async _handleBufferedUpload(req, res, next) {
236
+ const config = req.config;
237
+ const filesController = config.filesController;
238
+ const {
239
+ filename
240
+ } = req.params;
241
+ const contentType = req.get('Content-type');
242
+ if (!req.body || !req.body.length) {
243
+ next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, 'Invalid file upload.'));
244
+ return;
245
+ }
164
246
  const base64 = req.body.toString('base64');
165
247
  const file = new _node.default.File(filename, {
166
248
  base64
@@ -203,7 +285,12 @@ class FilesRouter {
203
285
  // if the file returned by the trigger has already been saved skip saving anything
204
286
  if (!saveResult) {
205
287
  // update fileSize
206
- const bufferData = Buffer.from(fileObject.file._data, 'base64');
288
+ let bufferData;
289
+ if (fileObject.file._source?.format === 'buffer') {
290
+ bufferData = fileObject.file._source.buffer;
291
+ } else {
292
+ bufferData = Buffer.from(fileObject.file._data, 'base64');
293
+ }
207
294
  fileObject.fileSize = Buffer.byteLength(bufferData);
208
295
  // prepare file options
209
296
  const fileOptions = {
@@ -241,6 +328,126 @@ class FilesRouter {
241
328
  next(error);
242
329
  }
243
330
  }
331
+ async _handleStreamUpload(req, res, next) {
332
+ const config = req.config;
333
+ const filesController = config.filesController;
334
+ const {
335
+ filename
336
+ } = req.params;
337
+ let contentType = req.get('Content-Type');
338
+ const maxBytes = req._maxUploadSizeBytes;
339
+ let stream;
340
+ try {
341
+ // Early rejection via Content-Length header
342
+ const contentLength = req.get('Content-Length');
343
+ if (contentLength && parseInt(contentLength, 10) > maxBytes) {
344
+ req.resume();
345
+ next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, `File size exceeds maximum allowed: ${maxBytes} bytes.`));
346
+ return;
347
+ }
348
+ const mime = (await import('mime')).default;
349
+
350
+ // Infer content type from extension or add extension from content type
351
+ const hasExtension = filename && filename.includes('.');
352
+ if (hasExtension && !contentType) {
353
+ contentType = mime.getType(filename);
354
+ } else if (!hasExtension && contentType) {
355
+ // extension will be added by filesController.createFile
356
+ }
357
+
358
+ // Create size-limited stream wrapping the request
359
+ stream = createSizeLimitedStream(req, maxBytes);
360
+
361
+ // Build a Parse.File with no _data (streaming mode)
362
+ const file = new _node.default.File(filename, {
363
+ base64: ''
364
+ }, contentType);
365
+ const {
366
+ metadata = {},
367
+ tags = {}
368
+ } = req.fileData || {};
369
+
370
+ // Validate metadata and tags for prohibited keywords
371
+ try {
372
+ Utils.checkProhibitedKeywords(config, metadata);
373
+ Utils.checkProhibitedKeywords(config, tags);
374
+ } catch (error) {
375
+ stream.destroy();
376
+ next(new _node.default.Error(_node.default.Error.INVALID_KEY_NAME, error));
377
+ return;
378
+ }
379
+ file.setTags(tags);
380
+ file.setMetadata(metadata);
381
+ const fileSize = req.get('Content-Length') ? parseInt(req.get('Content-Length'), 10) : null;
382
+ const fileObject = {
383
+ file,
384
+ fileSize,
385
+ stream: true
386
+ };
387
+
388
+ // Run beforeSaveFile trigger
389
+ const triggerResult = await triggers.maybeRunFileTrigger(triggers.Types.beforeSave, fileObject, config, req.auth);
390
+ let saveResult;
391
+ // If a new ParseFile is returned, check if it's an already saved file
392
+ if (triggerResult instanceof _node.default.File) {
393
+ fileObject.file = triggerResult;
394
+ if (triggerResult.url()) {
395
+ fileObject.fileSize = null;
396
+ saveResult = {
397
+ url: triggerResult.url(),
398
+ name: triggerResult._name
399
+ };
400
+ // Destroy stream to remove listeners and drain request
401
+ stream.destroy();
402
+ }
403
+ }
404
+
405
+ // If the file returned by the trigger has already been saved, skip saving
406
+ if (!saveResult) {
407
+ // Prepare file options
408
+ const fileOptions = {
409
+ metadata: fileObject.file._metadata
410
+ };
411
+ const fileTags = Object.keys(fileObject.file._tags).length > 0 ? {
412
+ tags: fileObject.file._tags
413
+ } : {};
414
+ Object.assign(fileOptions, fileTags);
415
+
416
+ // Pass stream directly to filesController — it will buffer if adapter doesn't support streaming
417
+ const sourceType = fileObject.file._source?.type || contentType;
418
+ const createFileResult = await filesController.createFile(config, fileObject.file._name, stream, sourceType, fileOptions);
419
+
420
+ // Update file with new data
421
+ fileObject.file._name = createFileResult.name;
422
+ fileObject.file._url = createFileResult.url;
423
+ fileObject.file._requestTask = null;
424
+ fileObject.file._previousSave = Promise.resolve(fileObject.file);
425
+ saveResult = {
426
+ url: createFileResult.url,
427
+ name: createFileResult.name
428
+ };
429
+ }
430
+
431
+ // Run afterSaveFile trigger
432
+ await triggers.maybeRunFileTrigger(triggers.Types.afterSave, fileObject, config, req.auth);
433
+ res.status(201);
434
+ res.set('Location', saveResult.url);
435
+ res.json(saveResult);
436
+ } catch (e) {
437
+ // Destroy stream to remove listeners and drain request, or resume directly
438
+ if (stream) {
439
+ stream.destroy();
440
+ } else {
441
+ req.resume();
442
+ }
443
+ _logger.default.error('Error creating a file: ', e);
444
+ const error = triggers.resolveError(e, {
445
+ code: _node.default.Error.FILE_SAVE_ERROR,
446
+ message: `Could not store file: ${filename}.`
447
+ });
448
+ next(error);
449
+ }
450
+ }
244
451
  async deleteHandler(req, res, next) {
245
452
  try {
246
453
  const {
@@ -298,4 +505,4 @@ function isFileStreamable(req, filesController) {
298
505
  const end = Number(range[1]);
299
506
  return (!isNaN(start) || !isNaN(end)) && typeof filesController.adapter.handleFileStream === 'function';
300
507
  }
301
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_express","_interopRequireDefault","require","Middlewares","_interopRequireWildcard","_node","_Config","_logger","e","t","WeakMap","r","n","__esModule","o","i","f","__proto__","default","has","get","set","hasOwnProperty","call","Object","defineProperty","getOwnPropertyDescriptor","triggers","Utils","FilesRouter","expressRouter","maxUploadSize","router","express","Router","getHandler","metadataHandler","post","req","res","next","Parse","Error","INVALID_FILE_NAME","raw","type","limit","handleParseHeaders","handleParseSession","createHandler","delete","enforceMasterKeyAccess","deleteHandler","config","Config","params","appId","status","json","code","OPERATION_FORBIDDEN","error","filename","filesController","mime","contentType","getType","file","File","base64","triggerResult","maybeRunFileTrigger","Types","beforeFind","auth","_name","isFileStreamable","handleFileStream","catch","end","data","getFileData","toString","afterFind","forceDownload","Buffer","from","_data","length","err","resolveError","SCRIPT_FAILED","message","user","isMaster","isLinked","AnonymousUtils","fileUpload","enableForAnonymousUser","FILE_SAVE_ERROR","enableForAuthenticatedUser","enableForPublic","body","validateFilename","fileExtensions","isValidExtension","extension","some","ext","regex","RegExp","test","includes","substring","lastIndexOf","split","join","metadata","tags","fileData","checkProhibitedKeywords","INVALID_KEY_NAME","setTags","setMetadata","fileSize","byteLength","fileObject","beforeSave","saveResult","url","name","bufferData","fileOptions","_metadata","fileTags","keys","_tags","assign","createFileResult","createFile","_source","_url","_requestTask","_previousSave","Promise","resolve","afterSave","logger","adapter","getFileLocation","beforeDelete","deleteFile","afterDelete","FILE_DELETE_ERROR","getMetadata","exports","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');\n\nexport class FilesRouter {\n  expressRouter({ maxUploadSize = '20Mb' } = {}) {\n    var router = express.Router();\n    router.get('/files/:appId/:filename', this.getHandler);\n    router.get('/files/:appId/metadata/:filename', this.metadataHandler);\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      express.raw({\n        type: () => {\n          return true;\n        },\n        limit: maxUploadSize,\n      }), // Allow uploads without Content-Type, or with any Content-Type.\n      Middlewares.handleParseHeaders,\n      Middlewares.handleParseSession,\n      this.createHandler\n    );\n\n    router.delete(\n      '/files/:filename',\n      Middlewares.handleParseHeaders,\n      Middlewares.handleParseSession,\n      Middlewares.enforceMasterKeyAccess,\n      this.deleteHandler\n    );\n    return router;\n  }\n\n  async getHandler(req, res) {\n    const config = Config.get(req.params.appId);\n    if (!config) {\n      res.status(403);\n      res.json({ code: Parse.Error.OPERATION_FORBIDDEN, error: 'Invalid application ID.' });\n      return;\n    }\n\n    let filename = req.params.filename;\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      if (isFileStreamable(req, filesController)) {\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 },\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      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  async createHandler(req, res, next) {\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    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 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      extension = extension?.split(' ')?.join('');\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    const base64 = req.body.toString('base64');\n    const file = new Parse.File(filename, { base64 }, contentType);\n    const { metadata = {}, tags = {} } = 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    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        const bufferData = Buffer.from(fileObject.file._data, 'base64');\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        // 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 deleteHandler(req, res, next) {\n    try {\n      const { filesController } = req.config;\n      const { filename } = req.params;\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      const { filesController } = config;\n      const { filename } = req.params;\n      const data = await filesController.getMetadata(filename);\n      res.status(200);\n      res.json(data);\n    } catch {\n      res.status(200);\n      res.json({});\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":";;;;;;AAAA,IAAAA,QAAA,GAAAC,sBAAA,CAAAC,OAAA;AACA,IAAAC,WAAA,GAAAC,uBAAA,CAAAF,OAAA;AACA,IAAAG,KAAA,GAAAJ,sBAAA,CAAAC,OAAA;AACA,IAAAI,OAAA,GAAAL,sBAAA,CAAAC,OAAA;AACA,IAAAK,OAAA,GAAAN,sBAAA,CAAAC,OAAA;AAA+B,SAAAE,wBAAAI,CAAA,EAAAC,CAAA,6BAAAC,OAAA,MAAAC,CAAA,OAAAD,OAAA,IAAAE,CAAA,OAAAF,OAAA,YAAAN,uBAAA,YAAAA,CAAAI,CAAA,EAAAC,CAAA,SAAAA,CAAA,IAAAD,CAAA,IAAAA,CAAA,CAAAK,UAAA,SAAAL,CAAA,MAAAM,CAAA,EAAAC,CAAA,EAAAC,CAAA,KAAAC,SAAA,QAAAC,OAAA,EAAAV,CAAA,iBAAAA,CAAA,uBAAAA,CAAA,yBAAAA,CAAA,SAAAQ,CAAA,MAAAF,CAAA,GAAAL,CAAA,GAAAG,CAAA,GAAAD,CAAA,QAAAG,CAAA,CAAAK,GAAA,CAAAX,CAAA,UAAAM,CAAA,CAAAM,GAAA,CAAAZ,CAAA,GAAAM,CAAA,CAAAO,GAAA,CAAAb,CAAA,EAAAQ,CAAA,gBAAAP,CAAA,IAAAD,CAAA,gBAAAC,CAAA,OAAAa,cAAA,CAAAC,IAAA,CAAAf,CAAA,EAAAC,CAAA,OAAAM,CAAA,IAAAD,CAAA,GAAAU,MAAA,CAAAC,cAAA,KAAAD,MAAA,CAAAE,wBAAA,CAAAlB,CAAA,EAAAC,CAAA,OAAAM,CAAA,CAAAK,GAAA,IAAAL,CAAA,CAAAM,GAAA,IAAAP,CAAA,CAAAE,CAAA,EAAAP,CAAA,EAAAM,CAAA,IAAAC,CAAA,CAAAP,CAAA,IAAAD,CAAA,CAAAC,CAAA,WAAAO,CAAA,KAAAR,CAAA,EAAAC,CAAA;AAAA,SAAAR,uBAAAO,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAK,UAAA,GAAAL,CAAA,KAAAU,OAAA,EAAAV,CAAA;AAC/B,MAAMmB,QAAQ,GAAGzB,OAAO,CAAC,aAAa,CAAC;AACvC,MAAM0B,KAAK,GAAG1B,OAAO,CAAC,UAAU,CAAC;AAE1B,MAAM2B,WAAW,CAAC;EACvBC,aAAaA,CAAC;IAAEC,aAAa,GAAG;EAAO,CAAC,GAAG,CAAC,CAAC,EAAE;IAC7C,IAAIC,MAAM,GAAGC,gBAAO,CAACC,MAAM,CAAC,CAAC;IAC7BF,MAAM,CAACZ,GAAG,CAAC,yBAAyB,EAAE,IAAI,CAACe,UAAU,CAAC;IACtDH,MAAM,CAACZ,GAAG,CAAC,kCAAkC,EAAE,IAAI,CAACgB,eAAe,CAAC;IAEpEJ,MAAM,CAACK,IAAI,CAAC,QAAQ,EAAE,UAAUC,GAAG,EAAEC,GAAG,EAAEC,IAAI,EAAE;MAC9CA,IAAI,CAAC,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACC,iBAAiB,EAAE,wBAAwB,CAAC,CAAC;IAChF,CAAC,CAAC;IAEFX,MAAM,CAACK,IAAI,CACT,kBAAkB,EAClBJ,gBAAO,CAACW,GAAG,CAAC;MACVC,IAAI,EAAEA,CAAA,KAAM;QACV,OAAO,IAAI;MACb,CAAC;MACDC,KAAK,EAAEf;IACT,CAAC,CAAC;IAAE;IACJ5B,WAAW,CAAC4C,kBAAkB,EAC9B5C,WAAW,CAAC6C,kBAAkB,EAC9B,IAAI,CAACC,aACP,CAAC;IAEDjB,MAAM,CAACkB,MAAM,CACX,kBAAkB,EAClB/C,WAAW,CAAC4C,kBAAkB,EAC9B5C,WAAW,CAAC6C,kBAAkB,EAC9B7C,WAAW,CAACgD,sBAAsB,EAClC,IAAI,CAACC,aACP,CAAC;IACD,OAAOpB,MAAM;EACf;EAEA,MAAMG,UAAUA,CAACG,GAAG,EAAEC,GAAG,EAAE;IACzB,MAAMc,MAAM,GAAGC,eAAM,CAAClC,GAAG,CAACkB,GAAG,CAACiB,MAAM,CAACC,KAAK,CAAC;IAC3C,IAAI,CAACH,MAAM,EAAE;MACXd,GAAG,CAACkB,MAAM,CAAC,GAAG,CAAC;MACflB,GAAG,CAACmB,IAAI,CAAC;QAAEC,IAAI,EAAElB,aAAK,CAACC,KAAK,CAACkB,mBAAmB;QAAEC,KAAK,EAAE;MAA0B,CAAC,CAAC;MACrF;IACF;IAEA,IAAIC,QAAQ,GAAGxB,GAAG,CAACiB,MAAM,CAACO,QAAQ;IAClC,IAAI;MACF,MAAMC,eAAe,GAAGV,MAAM,CAACU,eAAe;MAC9C,MAAMC,IAAI,GAAG,CAAC,MAAM,MAAM,CAAC,MAAM,CAAC,EAAE9C,OAAO;MAC3C,IAAI+C,WAAW,GAAGD,IAAI,CAACE,OAAO,CAACJ,QAAQ,CAAC;MACxC,IAAIK,IAAI,GAAG,IAAI1B,aAAK,CAAC2B,IAAI,CAACN,QAAQ,EAAE;QAAEO,MAAM,EAAE;MAAG,CAAC,EAAEJ,WAAW,CAAC;MAChE,MAAMK,aAAa,GAAG,MAAM3C,QAAQ,CAAC4C,mBAAmB,CACtD5C,QAAQ,CAAC6C,KAAK,CAACC,UAAU,EACzB;QAAEN;MAAK,CAAC,EACRd,MAAM,EACNf,GAAG,CAACoC,IACN,CAAC;MACD,IAAIJ,aAAa,EAAEH,IAAI,EAAEQ,KAAK,EAAE;QAC9Bb,QAAQ,GAAGQ,aAAa,EAAEH,IAAI,EAAEQ,KAAK;QACrCV,WAAW,GAAGD,IAAI,CAACE,OAAO,CAACJ,QAAQ,CAAC;MACtC;MAEA,IAAIc,gBAAgB,CAACtC,GAAG,EAAEyB,eAAe,CAAC,EAAE;QAC1CA,eAAe,CAACc,gBAAgB,CAACxB,MAAM,EAAES,QAAQ,EAAExB,GAAG,EAAEC,GAAG,EAAE0B,WAAW,CAAC,CAACa,KAAK,CAAC,MAAM;UACpFvC,GAAG,CAACkB,MAAM,CAAC,GAAG,CAAC;UACflB,GAAG,CAAClB,GAAG,CAAC,cAAc,EAAE,YAAY,CAAC;UACrCkB,GAAG,CAACwC,GAAG,CAAC,iBAAiB,CAAC;QAC5B,CAAC,CAAC;QACF;MACF;MAEA,IAAIC,IAAI,GAAG,MAAMjB,eAAe,CAACkB,WAAW,CAAC5B,MAAM,EAAES,QAAQ,CAAC,CAACgB,KAAK,CAAC,MAAM;QACzEvC,GAAG,CAACkB,MAAM,CAAC,GAAG,CAAC;QACflB,GAAG,CAAClB,GAAG,CAAC,cAAc,EAAE,YAAY,CAAC;QACrCkB,GAAG,CAACwC,GAAG,CAAC,iBAAiB,CAAC;MAC5B,CAAC,CAAC;MACF,IAAI,CAACC,IAAI,EAAE;QACT;MACF;MACAb,IAAI,GAAG,IAAI1B,aAAK,CAAC2B,IAAI,CAACN,QAAQ,EAAE;QAAEO,MAAM,EAAEW,IAAI,CAACE,QAAQ,CAAC,QAAQ;MAAE,CAAC,EAAEjB,WAAW,CAAC;MACjF,MAAMkB,SAAS,GAAG,MAAMxD,QAAQ,CAAC4C,mBAAmB,CAClD5C,QAAQ,CAAC6C,KAAK,CAACW,SAAS,EACxB;QAAEhB,IAAI;QAAEiB,aAAa,EAAE;MAAM,CAAC,EAC9B/B,MAAM,EACNf,GAAG,CAACoC,IACN,CAAC;MAED,IAAIS,SAAS,EAAEhB,IAAI,EAAE;QACnBF,WAAW,GAAGD,IAAI,CAACE,OAAO,CAACiB,SAAS,CAAChB,IAAI,CAACQ,KAAK,CAAC;QAChDK,IAAI,GAAGK,MAAM,CAACC,IAAI,CAACH,SAAS,CAAChB,IAAI,CAACoB,KAAK,EAAE,QAAQ,CAAC;MACpD;MAEAhD,GAAG,CAACkB,MAAM,CAAC,GAAG,CAAC;MACflB,GAAG,CAAClB,GAAG,CAAC,cAAc,EAAE4C,WAAW,CAAC;MACpC1B,GAAG,CAAClB,GAAG,CAAC,gBAAgB,EAAE2D,IAAI,CAACQ,MAAM,CAAC;MACtC,IAAIL,SAAS,CAACC,aAAa,EAAE;QAC3B7C,GAAG,CAAClB,GAAG,CAAC,qBAAqB,EAAE,uBAAuB8D,SAAS,CAAChB,IAAI,CAACQ,KAAK,EAAE,CAAC;MAC/E;MACApC,GAAG,CAACwC,GAAG,CAACC,IAAI,CAAC;IACf,CAAC,CAAC,OAAOxE,CAAC,EAAE;MACV,MAAMiF,GAAG,GAAG9D,QAAQ,CAAC+D,YAAY,CAAClF,CAAC,EAAE;QACnCmD,IAAI,EAAElB,aAAK,CAACC,KAAK,CAACiD,aAAa;QAC/BC,OAAO,EAAE,wBAAwB9B,QAAQ;MAC3C,CAAC,CAAC;MACFvB,GAAG,CAACkB,MAAM,CAAC,GAAG,CAAC;MACflB,GAAG,CAACmB,IAAI,CAAC;QAAEC,IAAI,EAAE8B,GAAG,CAAC9B,IAAI;QAAEE,KAAK,EAAE4B,GAAG,CAACG;MAAQ,CAAC,CAAC;IAClD;EACF;EAEA,MAAM3C,aAAaA,CAACX,GAAG,EAAEC,GAAG,EAAEC,IAAI,EAAE;IAClC,MAAMa,MAAM,GAAGf,GAAG,CAACe,MAAM;IACzB,MAAMwC,IAAI,GAAGvD,GAAG,CAACoC,IAAI,CAACmB,IAAI;IAC1B,MAAMC,QAAQ,GAAGxD,GAAG,CAACoC,IAAI,CAACoB,QAAQ;IAClC,MAAMC,QAAQ,GAAGF,IAAI,IAAIpD,aAAK,CAACuD,cAAc,CAACD,QAAQ,CAACF,IAAI,CAAC;IAC5D,IAAI,CAACC,QAAQ,IAAI,CAACzC,MAAM,CAAC4C,UAAU,CAACC,sBAAsB,IAAIH,QAAQ,EAAE;MACtEvD,IAAI,CACF,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACyD,eAAe,EAAE,4CAA4C,CAC3F,CAAC;MACD;IACF;IACA,IAAI,CAACL,QAAQ,IAAI,CAACzC,MAAM,CAAC4C,UAAU,CAACG,0BAA0B,IAAI,CAACL,QAAQ,IAAIF,IAAI,EAAE;MACnFrD,IAAI,CACF,IAAIC,aAAK,CAACC,KAAK,CACbD,aAAK,CAACC,KAAK,CAACyD,eAAe,EAC3B,gDACF,CACF,CAAC;MACD;IACF;IACA,IAAI,CAACL,QAAQ,IAAI,CAACzC,MAAM,CAAC4C,UAAU,CAACI,eAAe,IAAI,CAACR,IAAI,EAAE;MAC5DrD,IAAI,CAAC,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACyD,eAAe,EAAE,oCAAoC,CAAC,CAAC;MACxF;IACF;IACA,MAAMpC,eAAe,GAAGV,MAAM,CAACU,eAAe;IAC9C,MAAM;MAAED;IAAS,CAAC,GAAGxB,GAAG,CAACiB,MAAM;IAC/B,MAAMU,WAAW,GAAG3B,GAAG,CAAClB,GAAG,CAAC,cAAc,CAAC;IAE3C,IAAI,CAACkB,GAAG,CAACgE,IAAI,IAAI,CAAChE,GAAG,CAACgE,IAAI,CAACd,MAAM,EAAE;MACjChD,IAAI,CAAC,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACyD,eAAe,EAAE,sBAAsB,CAAC,CAAC;MAC1E;IACF;IAEA,MAAMtC,KAAK,GAAGE,eAAe,CAACwC,gBAAgB,CAACzC,QAAQ,CAAC;IACxD,IAAID,KAAK,EAAE;MACTrB,IAAI,CAACqB,KAAK,CAAC;MACX;IACF;IAEA,MAAM2C,cAAc,GAAGnD,MAAM,CAAC4C,UAAU,EAAEO,cAAc;IACxD,IAAI,CAACV,QAAQ,IAAIU,cAAc,EAAE;MAC/B,MAAMC,gBAAgB,GAAGC,SAAS,IAAI;QACpC,OAAOF,cAAc,CAACG,IAAI,CAACC,GAAG,IAAI;UAChC,IAAIA,GAAG,KAAK,GAAG,EAAE;YACf,OAAO,IAAI;UACb;UACA,MAAMC,KAAK,GAAG,IAAIC,MAAM,CAACF,GAAG,CAAC;UAC7B,IAAIC,KAAK,CAACE,IAAI,CAACL,SAAS,CAAC,EAAE;YACzB,OAAO,IAAI;UACb;QACF,CAAC,CAAC;MACJ,CAAC;MACD,IAAIA,SAAS,GAAGzC,WAAW;MAC3B,IAAIH,QAAQ,IAAIA,QAAQ,CAACkD,QAAQ,CAAC,GAAG,CAAC,EAAE;QACtCN,SAAS,GAAG5C,QAAQ,CAACmD,SAAS,CAACnD,QAAQ,CAACoD,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;MAC/D,CAAC,MAAM,IAAIjD,WAAW,IAAIA,WAAW,CAAC+C,QAAQ,CAAC,GAAG,CAAC,EAAE;QACnDN,SAAS,GAAGzC,WAAW,CAACkD,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;MACvC;MACAT,SAAS,GAAGA,SAAS,EAAES,KAAK,CAAC,GAAG,CAAC,EAAEC,IAAI,CAAC,EAAE,CAAC;MAE3C,IAAIV,SAAS,IAAI,CAACD,gBAAgB,CAACC,SAAS,CAAC,EAAE;QAC7ClE,IAAI,CACF,IAAIC,aAAK,CAACC,KAAK,CACbD,aAAK,CAACC,KAAK,CAACyD,eAAe,EAC3B,4BAA4BO,SAAS,eACvC,CACF,CAAC;QACD;MACF;IACF;IAEA,MAAMrC,MAAM,GAAG/B,GAAG,CAACgE,IAAI,CAACpB,QAAQ,CAAC,QAAQ,CAAC;IAC1C,MAAMf,IAAI,GAAG,IAAI1B,aAAK,CAAC2B,IAAI,CAACN,QAAQ,EAAE;MAAEO;IAAO,CAAC,EAAEJ,WAAW,CAAC;IAC9D,MAAM;MAAEoD,QAAQ,GAAG,CAAC,CAAC;MAAEC,IAAI,GAAG,CAAC;IAAE,CAAC,GAAGhF,GAAG,CAACiF,QAAQ,IAAI,CAAC,CAAC;IACvD,IAAI;MACF;MACA3F,KAAK,CAAC4F,uBAAuB,CAACnE,MAAM,EAAEgE,QAAQ,CAAC;MAC/CzF,KAAK,CAAC4F,uBAAuB,CAACnE,MAAM,EAAEiE,IAAI,CAAC;IAC7C,CAAC,CAAC,OAAOzD,KAAK,EAAE;MACdrB,IAAI,CAAC,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAAC+E,gBAAgB,EAAE5D,KAAK,CAAC,CAAC;MAC1D;IACF;IACAM,IAAI,CAACuD,OAAO,CAACJ,IAAI,CAAC;IAClBnD,IAAI,CAACwD,WAAW,CAACN,QAAQ,CAAC;IAC1B,MAAMO,QAAQ,GAAGvC,MAAM,CAACwC,UAAU,CAACvF,GAAG,CAACgE,IAAI,CAAC;IAC5C,MAAMwB,UAAU,GAAG;MAAE3D,IAAI;MAAEyD;IAAS,CAAC;IACrC,IAAI;MACF;MACA,MAAMtD,aAAa,GAAG,MAAM3C,QAAQ,CAAC4C,mBAAmB,CACtD5C,QAAQ,CAAC6C,KAAK,CAACuD,UAAU,EACzBD,UAAU,EACVzE,MAAM,EACNf,GAAG,CAACoC,IACN,CAAC;MACD,IAAIsD,UAAU;MACd;MACA,IAAI1D,aAAa,YAAY7B,aAAK,CAAC2B,IAAI,EAAE;QACvC0D,UAAU,CAAC3D,IAAI,GAAGG,aAAa;QAC/B,IAAIA,aAAa,CAAC2D,GAAG,CAAC,CAAC,EAAE;UACvB;UACAH,UAAU,CAACF,QAAQ,GAAG,IAAI;UAC1BI,UAAU,GAAG;YACXC,GAAG,EAAE3D,aAAa,CAAC2D,GAAG,CAAC,CAAC;YACxBC,IAAI,EAAE5D,aAAa,CAACK;UACtB,CAAC;QACH;MACF;MACA;MACA,IAAI,CAACqD,UAAU,EAAE;QACf;QACA,MAAMG,UAAU,GAAG9C,MAAM,CAACC,IAAI,CAACwC,UAAU,CAAC3D,IAAI,CAACoB,KAAK,EAAE,QAAQ,CAAC;QAC/DuC,UAAU,CAACF,QAAQ,GAAGvC,MAAM,CAACwC,UAAU,CAACM,UAAU,CAAC;QACnD;QACA,MAAMC,WAAW,GAAG;UAClBf,QAAQ,EAAES,UAAU,CAAC3D,IAAI,CAACkE;QAC5B,CAAC;QACD;QACA;QACA,MAAMC,QAAQ,GACZ9G,MAAM,CAAC+G,IAAI,CAACT,UAAU,CAAC3D,IAAI,CAACqE,KAAK,CAAC,CAAChD,MAAM,GAAG,CAAC,GAAG;UAAE8B,IAAI,EAAEQ,UAAU,CAAC3D,IAAI,CAACqE;QAAM,CAAC,GAAG,CAAC,CAAC;QACtFhH,MAAM,CAACiH,MAAM,CAACL,WAAW,EAAEE,QAAQ,CAAC;QACpC;QACA,MAAMI,gBAAgB,GAAG,MAAM3E,eAAe,CAAC4E,UAAU,CACvDtF,MAAM,EACNyE,UAAU,CAAC3D,IAAI,CAACQ,KAAK,EACrBwD,UAAU,EACVL,UAAU,CAAC3D,IAAI,CAACyE,OAAO,CAAC/F,IAAI,EAC5BuF,WACF,CAAC;QACD;QACAN,UAAU,CAAC3D,IAAI,CAACQ,KAAK,GAAG+D,gBAAgB,CAACR,IAAI;QAC7CJ,UAAU,CAAC3D,IAAI,CAAC0E,IAAI,GAAGH,gBAAgB,CAACT,GAAG;QAC3CH,UAAU,CAAC3D,IAAI,CAAC2E,YAAY,GAAG,IAAI;QACnChB,UAAU,CAAC3D,IAAI,CAAC4E,aAAa,GAAGC,OAAO,CAACC,OAAO,CAACnB,UAAU,CAAC3D,IAAI,CAAC;QAChE6D,UAAU,GAAG;UACXC,GAAG,EAAES,gBAAgB,CAACT,GAAG;UACzBC,IAAI,EAAEQ,gBAAgB,CAACR;QACzB,CAAC;MACH;MACA;MACA,MAAMvG,QAAQ,CAAC4C,mBAAmB,CAAC5C,QAAQ,CAAC6C,KAAK,CAAC0E,SAAS,EAAEpB,UAAU,EAAEzE,MAAM,EAAEf,GAAG,CAACoC,IAAI,CAAC;MAC1FnC,GAAG,CAACkB,MAAM,CAAC,GAAG,CAAC;MACflB,GAAG,CAAClB,GAAG,CAAC,UAAU,EAAE2G,UAAU,CAACC,GAAG,CAAC;MACnC1F,GAAG,CAACmB,IAAI,CAACsE,UAAU,CAAC;IACtB,CAAC,CAAC,OAAOxH,CAAC,EAAE;MACV2I,eAAM,CAACtF,KAAK,CAAC,yBAAyB,EAAErD,CAAC,CAAC;MAC1C,MAAMqD,KAAK,GAAGlC,QAAQ,CAAC+D,YAAY,CAAClF,CAAC,EAAE;QACrCmD,IAAI,EAAElB,aAAK,CAACC,KAAK,CAACyD,eAAe;QACjCP,OAAO,EAAE,yBAAyBkC,UAAU,CAAC3D,IAAI,CAACQ,KAAK;MACzD,CAAC,CAAC;MACFnC,IAAI,CAACqB,KAAK,CAAC;IACb;EACF;EAEA,MAAMT,aAAaA,CAACd,GAAG,EAAEC,GAAG,EAAEC,IAAI,EAAE;IAClC,IAAI;MACF,MAAM;QAAEuB;MAAgB,CAAC,GAAGzB,GAAG,CAACe,MAAM;MACtC,MAAM;QAAES;MAAS,CAAC,GAAGxB,GAAG,CAACiB,MAAM;MAC/B;MACA,MAAMY,IAAI,GAAG,IAAI1B,aAAK,CAAC2B,IAAI,CAACN,QAAQ,CAAC;MACrCK,IAAI,CAAC0E,IAAI,GAAG,MAAM9E,eAAe,CAACqF,OAAO,CAACC,eAAe,CAAC/G,GAAG,CAACe,MAAM,EAAES,QAAQ,CAAC;MAC/E,MAAMgE,UAAU,GAAG;QAAE3D,IAAI;QAAEyD,QAAQ,EAAE;MAAK,CAAC;MAC3C,MAAMjG,QAAQ,CAAC4C,mBAAmB,CAChC5C,QAAQ,CAAC6C,KAAK,CAAC8E,YAAY,EAC3BxB,UAAU,EACVxF,GAAG,CAACe,MAAM,EACVf,GAAG,CAACoC,IACN,CAAC;MACD;MACA,MAAMX,eAAe,CAACwF,UAAU,CAACjH,GAAG,CAACe,MAAM,EAAES,QAAQ,CAAC;MACtD;MACA,MAAMnC,QAAQ,CAAC4C,mBAAmB,CAChC5C,QAAQ,CAAC6C,KAAK,CAACgF,WAAW,EAC1B1B,UAAU,EACVxF,GAAG,CAACe,MAAM,EACVf,GAAG,CAACoC,IACN,CAAC;MACDnC,GAAG,CAACkB,MAAM,CAAC,GAAG,CAAC;MACf;MACAlB,GAAG,CAACwC,GAAG,CAAC,CAAC;IACX,CAAC,CAAC,OAAOvE,CAAC,EAAE;MACV2I,eAAM,CAACtF,KAAK,CAAC,yBAAyB,EAAErD,CAAC,CAAC;MAC1C,MAAMqD,KAAK,GAAGlC,QAAQ,CAAC+D,YAAY,CAAClF,CAAC,EAAE;QACrCmD,IAAI,EAAElB,aAAK,CAACC,KAAK,CAAC+G,iBAAiB;QACnC7D,OAAO,EAAE;MACX,CAAC,CAAC;MACFpD,IAAI,CAACqB,KAAK,CAAC;IACb;EACF;EAEA,MAAMzB,eAAeA,CAACE,GAAG,EAAEC,GAAG,EAAE;IAC9B,IAAI;MACF,MAAMc,MAAM,GAAGC,eAAM,CAAClC,GAAG,CAACkB,GAAG,CAACiB,MAAM,CAACC,KAAK,CAAC;MAC3C,MAAM;QAAEO;MAAgB,CAAC,GAAGV,MAAM;MAClC,MAAM;QAAES;MAAS,CAAC,GAAGxB,GAAG,CAACiB,MAAM;MAC/B,MAAMyB,IAAI,GAAG,MAAMjB,eAAe,CAAC2F,WAAW,CAAC5F,QAAQ,CAAC;MACxDvB,GAAG,CAACkB,MAAM,CAAC,GAAG,CAAC;MACflB,GAAG,CAACmB,IAAI,CAACsB,IAAI,CAAC;IAChB,CAAC,CAAC,MAAM;MACNzC,GAAG,CAACkB,MAAM,CAAC,GAAG,CAAC;MACflB,GAAG,CAACmB,IAAI,CAAC,CAAC,CAAC,CAAC;IACd;EACF;AACF;AAACiG,OAAA,CAAA9H,WAAA,GAAAA,WAAA;AAED,SAAS+C,gBAAgBA,CAACtC,GAAG,EAAEyB,eAAe,EAAE;EAC9C,MAAM6F,KAAK,GAAG,CAACtH,GAAG,CAAClB,GAAG,CAAC,OAAO,CAAC,IAAI,KAAK,EAAE+F,KAAK,CAAC,GAAG,CAAC;EACpD,MAAM0C,KAAK,GAAGC,MAAM,CAACF,KAAK,CAAC,CAAC,CAAC,CAAC;EAC9B,MAAM7E,GAAG,GAAG+E,MAAM,CAACF,KAAK,CAAC,CAAC,CAAC,CAAC;EAC5B,OACE,CAAC,CAACG,KAAK,CAACF,KAAK,CAAC,IAAI,CAACE,KAAK,CAAChF,GAAG,CAAC,KAAK,OAAOhB,eAAe,CAACqF,OAAO,CAACvE,gBAAgB,KAAK,UAAU;AAEpG","ignoreList":[]}
508
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_express","_interopRequireDefault","require","Middlewares","_interopRequireWildcard","_node","_Config","_logger","_stream","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","FilesRouter","expressRouter","maxUploadSize","router","express","Router","getHandler","metadataHandler","post","req","res","next","INVALID_FILE_NAME","_bodyParsingMiddleware","handleParseHeaders","handleParseSession","createHandler","bind","delete","enforceMasterKeyAccess","deleteHandler","config","Config","params","appId","status","json","code","OPERATION_FORBIDDEN","error","filename","filesController","mime","contentType","getType","file","File","base64","triggerResult","maybeRunFileTrigger","Types","beforeFind","auth","_name","isFileStreamable","handleFileStream","catch","end","data","getFileData","toString","afterFind","forceDownload","Buffer","from","_data","resolveError","SCRIPT_FAILED","message","rawParser","raw","type","limit","_maxUploadSizeBytes","parseSizeToBytes","user","isMaster","isLinked","AnonymousUtils","fileUpload","enableForAnonymousUser","enableForAuthenticatedUser","enableForPublic","validateFilename","fileExtensions","isValidExtension","extension","some","ext","regex","RegExp","test","includes","substring","lastIndexOf","split","join","body","_handleBufferedUpload","_handleStreamUpload","metadata","tags","fileData","checkProhibitedKeywords","INVALID_KEY_NAME","setTags","setMetadata","fileSize","byteLength","fileObject","beforeSave","saveResult","url","name","bufferData","_source","format","buffer","fileOptions","_metadata","fileTags","keys","_tags","assign","createFileResult","createFile","_url","_requestTask","_previousSave","Promise","resolve","afterSave","logger","stream","contentLength","parseInt","hasExtension","sourceType","adapter","getFileLocation","beforeDelete","deleteFile","afterDelete","FILE_DELETE_ERROR","getMetadata","exports","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';\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\nexport class FilesRouter {\n  expressRouter({ maxUploadSize = '20Mb' } = {}) {\n    var router = express.Router();\n    router.get('/files/:appId/:filename', this.getHandler);\n    router.get('/files/:appId/metadata/:filename', this.metadataHandler);\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._bodyParsingMiddleware(maxUploadSize),\n      Middlewares.handleParseHeaders,\n      Middlewares.handleParseSession,\n      this.createHandler.bind(this)\n    );\n\n    router.delete(\n      '/files/:filename',\n      Middlewares.handleParseHeaders,\n      Middlewares.handleParseSession,\n      Middlewares.enforceMasterKeyAccess,\n      this.deleteHandler\n    );\n    return router;\n  }\n\n  async getHandler(req, res) {\n    const config = Config.get(req.params.appId);\n    if (!config) {\n      res.status(403);\n      res.json({ code: Parse.Error.OPERATION_FORBIDDEN, error: 'Invalid application ID.' });\n      return;\n    }\n\n    let filename = req.params.filename;\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      if (isFileStreamable(req, filesController)) {\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 },\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      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  _bodyParsingMiddleware(maxUploadSize) {\n    const rawParser = express.raw({\n      type: () => true,\n      limit: maxUploadSize,\n    });\n    return (req, res, next) => {\n      if (req.get('X-Parse-Upload-Mode') === 'stream') {\n        req._maxUploadSizeBytes = Utils.parseSizeToBytes(maxUploadSize);\n        return next();\n      }\n      return rawParser(req, res, next);\n    };\n  }\n\n  async createHandler(req, res, next) {\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      extension = extension?.split(' ')?.join('');\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    // Dispatch to the appropriate handler based on whether the body was buffered\n    if (req.body instanceof Buffer) {\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 = {} } = 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    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        // 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 = {} } = 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\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\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    try {\n      const { filesController } = req.config;\n      const { filename } = req.params;\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      const { filesController } = config;\n      const { filename } = req.params;\n      const data = await filesController.getMetadata(filename);\n      res.status(200);\n      res.json(data);\n    } catch {\n      res.status(200);\n      res.json({});\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/Utils.js CHANGED
@@ -463,6 +463,40 @@ class Utils {
463
463
  }
464
464
  return current;
465
465
  }
466
+
467
+ /**
468
+ * Parses a human-readable size string into a byte count.
469
+ * @param {number | string} size - A number (floored to an integer), a numeric string
470
+ * (treated as bytes), or a string with a unit suffix: `b`, `kb`, `mb`, `gb`
471
+ * (case-insensitive). Examples: `'20mb'`, `'512kb'`, `'1.5gb'`, `1048576`.
472
+ * @returns {number} The size in bytes, floored to the nearest integer.
473
+ * @throws {Error} If the string does not match the expected format.
474
+ */
475
+ static parseSizeToBytes(size) {
476
+ if (typeof size === 'number') {
477
+ if (!Number.isFinite(size) || size < 0) {
478
+ throw new Error(`Invalid size value: ${size}`);
479
+ }
480
+ return Math.floor(size);
481
+ }
482
+ const str = String(size).trim().toLowerCase();
483
+ const match = str.match(/^(\d+(?:\.\d+)?)\s*(b|kb|mb|gb)?$/);
484
+ if (!match) {
485
+ throw new Error(`Invalid size value: ${size}`);
486
+ }
487
+ const num = parseFloat(match[1]);
488
+ const unit = match[2];
489
+ switch (unit) {
490
+ case 'kb':
491
+ return Math.floor(num * 1024);
492
+ case 'mb':
493
+ return Math.floor(num * 1024 * 1024);
494
+ case 'gb':
495
+ return Math.floor(num * 1024 * 1024 * 1024);
496
+ default:
497
+ return Math.floor(num);
498
+ }
499
+ }
466
500
  }
467
501
  module.exports = Utils;
468
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["path","require","fs","promises","Utils","getLocalizedPath","defaultPath","locale","file","basename","basePath","dirname","localePath","join","localeFileExists","fileExists","subdir","language","split","languagePath","languageFileExists","access","isPath","s","test","flattenObject","obj","parentKey","delimiter","result","key","Object","prototype","hasOwnProperty","call","newKey","isPromise","object","Promise","getObjectKeyPermutations","index","current","results","keys","values","value","nextIndex","length","assign","push","validateParams","params","types","type","isOptional","o","param","v","t","relativeTimeToDate","text","now","Date","toLowerCase","parts","filter","part","future","past","status","info","slice","pairs","shift","seconds","num","interval","val","Number","isInteger","milliseconds","valueOf","objectContainsKeyValue","isMatch","a","b","RegExp","isKeyMatch","k","isValueMatch","entries","undefined","includes","toString","checkProhibitedKeywords","config","data","requestKeywordDenylist","keyword","match","JSON","stringify","addNestedKeysToRoot","encodeForUrl","input","encodeURIComponent","replace","char","charCodeAt","toUpperCase","getCircularReplacer","seen","WeakSet","Map","fromEntries","Set","Array","from","has","add","getNestedProperty","module","exports"],"sources":["../src/Utils.js"],"sourcesContent":["/**\n * utils.js\n * @file General purpose utilities\n * @description General purpose utilities.\n */\n\nconst path = require('path');\nconst fs = require('fs').promises;\n\n/**\n * The general purpose utilities.\n */\nclass Utils {\n  /**\n   * @function getLocalizedPath\n   * @description Returns a localized file path accoring to the locale.\n   *\n   * Localized files are searched in subfolders of a given path, e.g.\n   *\n   * root/\n   * ├── base/                    // base path to files\n   * │   ├── example.html         // default file\n   * │   └── de/                  // de language folder\n   * │   │   └── example.html     // de localized file\n   * │   └── de-AT/               // de-AT locale folder\n   * │   │   └── example.html     // de-AT localized file\n   *\n   * Files are matched with the locale in the following order:\n   * 1. Locale match, e.g. locale `de-AT` matches file in folder `de-AT`.\n   * 2. Language match, e.g. locale `de-AT` matches file in folder `de`.\n   * 3. Default; file in base folder is returned.\n   *\n   * @param {String} defaultPath The absolute file path, which is also\n   * the default path returned if localization is not available.\n   * @param {String} locale The locale.\n   * @returns {Promise<Object>} The object contains:\n   * - `path`: The path to the localized file, or the original path if\n   *   localization is not available.\n   * - `subdir`: The subdirectory of the localized file, or undefined if\n   *   there is no matching localized file.\n   */\n  static async getLocalizedPath(defaultPath, locale) {\n    // Get file name and paths\n    const file = path.basename(defaultPath);\n    const basePath = path.dirname(defaultPath);\n\n    // If locale is not set return default file\n    if (!locale) {\n      return { path: defaultPath };\n    }\n\n    // Check file for locale exists\n    const localePath = path.join(basePath, locale, file);\n    const localeFileExists = await Utils.fileExists(localePath);\n\n    // If file for locale exists return file\n    if (localeFileExists) {\n      return { path: localePath, subdir: locale };\n    }\n\n    // Check file for language exists\n    const language = locale.split('-')[0];\n    const languagePath = path.join(basePath, language, file);\n    const languageFileExists = await Utils.fileExists(languagePath);\n\n    // If file for language exists return file\n    if (languageFileExists) {\n      return { path: languagePath, subdir: language };\n    }\n\n    // Return default file\n    return { path: defaultPath };\n  }\n\n  /**\n   * @function fileExists\n   * @description Checks whether a file exists.\n   * @param {String} path The file path.\n   * @returns {Promise<Boolean>} Is true if the file can be accessed, false otherwise.\n   */\n  static async fileExists(path) {\n    try {\n      await fs.access(path);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * @function isPath\n   * @description Evaluates whether a string is a file path (as opposed to a URL for example).\n   * @param {String} s The string to evaluate.\n   * @returns {Boolean} Returns true if the evaluated string is a path.\n   */\n  static isPath(s) {\n    return /(^\\/)|(^\\.\\/)|(^\\.\\.\\/)/.test(s);\n  }\n\n  /**\n   * Flattens an object and crates new keys with custom delimiters.\n   * @param {Object} obj The object to flatten.\n   * @param {String} [delimiter='.'] The delimiter of the newly generated keys.\n   * @param {Object} result\n   * @returns {Object} The flattened object.\n   **/\n  static flattenObject(obj, parentKey, delimiter = '.', result = {}) {\n    for (const key in obj) {\n      if (Object.prototype.hasOwnProperty.call(obj, key)) {\n        const newKey = parentKey ? parentKey + delimiter + key : key;\n\n        if (typeof obj[key] === 'object' && obj[key] !== null) {\n          this.flattenObject(obj[key], newKey, delimiter, result);\n        } else {\n          result[newKey] = obj[key];\n        }\n      }\n    }\n    return result;\n  }\n\n  /**\n   * Determines whether an object is a Promise.\n   * @param {any} object The object to validate.\n   * @returns {Boolean} Returns true if the object is a promise.\n   */\n  static isPromise(object) {\n    return object instanceof Promise;\n  }\n\n  /**\n   * Creates an object with all permutations of the original keys.\n   * For example, this definition:\n   * ```\n   * {\n   *   a: [true, false],\n   *   b: [1, 2],\n   *   c: ['x']\n   * }\n   * ```\n   * permutates to:\n   * ```\n   * [\n   *   { a: true, b: 1, c: 'x' },\n   *   { a: true, b: 2, c: 'x' },\n   *   { a: false, b: 1, c: 'x' },\n   *   { a: false, b: 2, c: 'x' }\n   * ]\n   * ```\n   * @param {Object} object The object to permutate.\n   * @param {Integer} [index=0] The current key index.\n   * @param {Object} [current={}] The current result entry being composed.\n   * @param {Array} [results=[]] The resulting array of permutations.\n   */\n  static getObjectKeyPermutations(object, index = 0, current = {}, results = []) {\n    const keys = Object.keys(object);\n    const key = keys[index];\n    const values = object[key];\n\n    for (const value of values) {\n      current[key] = value;\n      const nextIndex = index + 1;\n\n      if (nextIndex < keys.length) {\n        Utils.getObjectKeyPermutations(object, nextIndex, current, results);\n      } else {\n        const result = Object.assign({}, current);\n        results.push(result);\n      }\n    }\n    return results;\n  }\n\n  /**\n   * Validates parameters and throws if a parameter is invalid.\n   * Example parameter types syntax:\n   * ```\n   * {\n   *   parameterName: {\n   *      t: 'boolean',\n   *      v: isBoolean,\n   *      o: true\n   *   },\n   *   ...\n   * }\n   * ```\n   * @param {Object} params The parameters to validate.\n   * @param {Array<Object>} types The parameter types used for validation.\n   * @param {Object} types.t The parameter type; used for error message, not for validation.\n   * @param {Object} types.v The function to validate the parameter value.\n   * @param {Boolean} [types.o=false] Is true if the parameter is optional.\n   */\n  static validateParams(params, types) {\n    for (const key of Object.keys(params)) {\n      const type = types[key];\n      const isOptional = !!type.o;\n      const param = params[key];\n      if (!(isOptional && param == null) && !type.v(param)) {\n        throw `Invalid parameter ${key} must be of type ${type.t} but is ${typeof param}`;\n      }\n    }\n  }\n\n  /**\n   * Computes the relative date based on a string.\n   * @param {String} text The string to interpret the date from.\n   * @param {Date} now The date the string is comparing against.\n   * @returns {Object} The relative date object.\n   **/\n  static relativeTimeToDate(text, now = new Date()) {\n    text = text.toLowerCase();\n    let parts = text.split(' ');\n\n    // Filter out whitespace\n    parts = parts.filter(part => part !== '');\n\n    const future = parts[0] === 'in';\n    const past = parts[parts.length - 1] === 'ago';\n\n    if (!future && !past && text !== 'now') {\n      return {\n        status: 'error',\n        info: \"Time should either start with 'in' or end with 'ago'\",\n      };\n    }\n\n    if (future && past) {\n      return {\n        status: 'error',\n        info: \"Time cannot have both 'in' and 'ago'\",\n      };\n    }\n\n    // strip the 'ago' or 'in'\n    if (future) {\n      parts = parts.slice(1);\n    } else {\n      // past\n      parts = parts.slice(0, parts.length - 1);\n    }\n\n    if (parts.length % 2 !== 0 && text !== 'now') {\n      return {\n        status: 'error',\n        info: 'Invalid time string. Dangling unit or number.',\n      };\n    }\n\n    const pairs = [];\n    while (parts.length) {\n      pairs.push([parts.shift(), parts.shift()]);\n    }\n\n    let seconds = 0;\n    for (const [num, interval] of pairs) {\n      const val = Number(num);\n      if (!Number.isInteger(val)) {\n        return {\n          status: 'error',\n          info: `'${num}' is not an integer.`,\n        };\n      }\n\n      switch (interval) {\n        case 'yr':\n        case 'yrs':\n        case 'year':\n        case 'years':\n          seconds += val * 31536000; // 365 * 24 * 60 * 60\n          break;\n\n        case 'wk':\n        case 'wks':\n        case 'week':\n        case 'weeks':\n          seconds += val * 604800; // 7 * 24 * 60 * 60\n          break;\n\n        case 'd':\n        case 'day':\n        case 'days':\n          seconds += val * 86400; // 24 * 60 * 60\n          break;\n\n        case 'hr':\n        case 'hrs':\n        case 'hour':\n        case 'hours':\n          seconds += val * 3600; // 60 * 60\n          break;\n\n        case 'min':\n        case 'mins':\n        case 'minute':\n        case 'minutes':\n          seconds += val * 60;\n          break;\n\n        case 'sec':\n        case 'secs':\n        case 'second':\n        case 'seconds':\n          seconds += val;\n          break;\n\n        default:\n          return {\n            status: 'error',\n            info: `Invalid interval: '${interval}'`,\n          };\n      }\n    }\n\n    const milliseconds = seconds * 1000;\n    if (future) {\n      return {\n        status: 'success',\n        info: 'future',\n        result: new Date(now.valueOf() + milliseconds),\n      };\n    } else if (past) {\n      return {\n        status: 'success',\n        info: 'past',\n        result: new Date(now.valueOf() - milliseconds),\n      };\n    } else {\n      return {\n        status: 'success',\n        info: 'present',\n        result: new Date(now.valueOf()),\n      };\n    }\n  }\n\n  /**\n   * Deep-scans an object for a matching key/value definition.\n   * @param {Object} obj The object to scan.\n   * @param {String | undefined} key The key to match, or undefined if only the value should be matched.\n   * @param {any | undefined} value The value to match, or undefined if only the key should be matched.\n   * @returns {Boolean} True if a match was found, false otherwise.\n   */\n  static objectContainsKeyValue(obj, key, value) {\n    const isMatch = (a, b) => (typeof a === 'string' && new RegExp(b).test(a)) || a === b;\n    const isKeyMatch = k => isMatch(k, key);\n    const isValueMatch = v => isMatch(v, value);\n    for (const [k, v] of Object.entries(obj)) {\n      if (key !== undefined && value === undefined && isKeyMatch(k)) {\n        return true;\n      } else if (key === undefined && value !== undefined && isValueMatch(v)) {\n        return true;\n      } else if (key !== undefined && value !== undefined && isKeyMatch(k) && isValueMatch(v)) {\n        return true;\n      }\n      if (['[object Object]', '[object Array]'].includes(Object.prototype.toString.call(v))) {\n        return Utils.objectContainsKeyValue(v, key, value);\n      }\n    }\n    return false;\n  }\n\n  static checkProhibitedKeywords(config, data) {\n    if (config?.requestKeywordDenylist) {\n      // Scan request data for denied keywords\n      for (const keyword of config.requestKeywordDenylist) {\n        const match = Utils.objectContainsKeyValue(data, keyword.key, keyword.value);\n        if (match) {\n          throw `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`;\n        }\n      }\n    }\n  }\n\n  /**\n   * Moves the nested keys of a specified key in an object to the root of the object.\n   *\n   * @param {Object} obj The object to modify.\n   * @param {String} key The key whose nested keys will be moved to root.\n   * @returns {Object} The modified object, or the original object if no modification happened.\n   * @example\n   * const obj = {\n   *   a: 1,\n   *   b: {\n   *     c: 2,\n   *     d: 3\n   *   },\n   *   e: 4\n   * };\n   * addNestedKeysToRoot(obj, 'b');\n   * console.log(obj);\n   * // Output: { a: 1, e: 4, c: 2, d: 3 }\n  */\n  static addNestedKeysToRoot(obj, key) {\n    if (obj[key] && typeof obj[key] === 'object') {\n      // Add nested keys to root\n      Object.assign(obj, { ...obj[key] });\n      // Delete original nested key\n      delete obj[key];\n    }\n    return obj;\n  }\n\n  /**\n   * Encodes a string to be used in a URL.\n   * @param {String} input The string to encode.\n   * @returns {String} The encoded string.\n   */\n  static encodeForUrl(input) {\n    return encodeURIComponent(input).replace(/[!'.()*]/g, char =>\n      '%' + char.charCodeAt(0).toString(16).toUpperCase()\n    );\n  }\n\n  /**\n   * Creates a JSON replacer function that handles Map, Set, and circular references.\n   * This replacer can be used with JSON.stringify to safely serialize complex objects.\n   *\n   * @returns {Function} A replacer function for JSON.stringify that:\n   * - Converts Map instances to plain objects\n   * - Converts Set instances to arrays\n   * - Replaces circular references with '[Circular]' marker\n   *\n   * @example\n   * const obj = { name: 'test', map: new Map([['key', 'value']]) };\n   * obj.self = obj; // circular reference\n   * JSON.stringify(obj, Utils.getCircularReplacer());\n   * // Output: {\"name\":\"test\",\"map\":{\"key\":\"value\"},\"self\":\"[Circular]\"}\n   */\n  static getCircularReplacer() {\n    const seen = new WeakSet();\n    return (key, value) => {\n      if (value instanceof Map) {\n        return Object.fromEntries(value);\n      }\n      if (value instanceof Set) {\n        return Array.from(value);\n      }\n      if (typeof value === 'object' && value !== null) {\n        if (seen.has(value)) {\n          return '[Circular]';\n        }\n        seen.add(value);\n      }\n      return value;\n    };\n  }\n\n  /**\n   * Gets a nested property value from an object using dot notation.\n   * @param {Object} obj The object to get the property from.\n   * @param {String} path The property path in dot notation, e.g. 'databaseOptions.allowPublicExplain'.\n   * @returns {any} The property value or undefined if not found.\n   * @example\n   * const obj = { database: { options: { enabled: true } } };\n   * Utils.getNestedProperty(obj, 'database.options.enabled');\n   * // Output: true\n   */\n  static getNestedProperty(obj, path) {\n    if (!obj || !path) {\n      return undefined;\n    }\n    const keys = path.split('.');\n    let current = obj;\n    for (const key of keys) {\n      if (current == null || typeof current !== 'object') {\n        return undefined;\n      }\n      current = current[key];\n    }\n    return current;\n  }\n}\n\nmodule.exports = Utils;\n"],"mappings":"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","ignoreList":[]}
502
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["path","require","fs","promises","Utils","getLocalizedPath","defaultPath","locale","file","basename","basePath","dirname","localePath","join","localeFileExists","fileExists","subdir","language","split","languagePath","languageFileExists","access","isPath","s","test","flattenObject","obj","parentKey","delimiter","result","key","Object","prototype","hasOwnProperty","call","newKey","isPromise","object","Promise","getObjectKeyPermutations","index","current","results","keys","values","value","nextIndex","length","assign","push","validateParams","params","types","type","isOptional","o","param","v","t","relativeTimeToDate","text","now","Date","toLowerCase","parts","filter","part","future","past","status","info","slice","pairs","shift","seconds","num","interval","val","Number","isInteger","milliseconds","valueOf","objectContainsKeyValue","isMatch","a","b","RegExp","isKeyMatch","k","isValueMatch","entries","undefined","includes","toString","checkProhibitedKeywords","config","data","requestKeywordDenylist","keyword","match","JSON","stringify","addNestedKeysToRoot","encodeForUrl","input","encodeURIComponent","replace","char","charCodeAt","toUpperCase","getCircularReplacer","seen","WeakSet","Map","fromEntries","Set","Array","from","has","add","getNestedProperty","parseSizeToBytes","size","isFinite","Error","Math","floor","str","String","trim","parseFloat","unit","module","exports"],"sources":["../src/Utils.js"],"sourcesContent":["/**\n * utils.js\n * @file General purpose utilities\n * @description General purpose utilities.\n */\n\nconst path = require('path');\nconst fs = require('fs').promises;\n\n/**\n * The general purpose utilities.\n */\nclass Utils {\n  /**\n   * @function getLocalizedPath\n   * @description Returns a localized file path accoring to the locale.\n   *\n   * Localized files are searched in subfolders of a given path, e.g.\n   *\n   * root/\n   * ├── base/                    // base path to files\n   * │   ├── example.html         // default file\n   * │   └── de/                  // de language folder\n   * │   │   └── example.html     // de localized file\n   * │   └── de-AT/               // de-AT locale folder\n   * │   │   └── example.html     // de-AT localized file\n   *\n   * Files are matched with the locale in the following order:\n   * 1. Locale match, e.g. locale `de-AT` matches file in folder `de-AT`.\n   * 2. Language match, e.g. locale `de-AT` matches file in folder `de`.\n   * 3. Default; file in base folder is returned.\n   *\n   * @param {String} defaultPath The absolute file path, which is also\n   * the default path returned if localization is not available.\n   * @param {String} locale The locale.\n   * @returns {Promise<Object>} The object contains:\n   * - `path`: The path to the localized file, or the original path if\n   *   localization is not available.\n   * - `subdir`: The subdirectory of the localized file, or undefined if\n   *   there is no matching localized file.\n   */\n  static async getLocalizedPath(defaultPath, locale) {\n    // Get file name and paths\n    const file = path.basename(defaultPath);\n    const basePath = path.dirname(defaultPath);\n\n    // If locale is not set return default file\n    if (!locale) {\n      return { path: defaultPath };\n    }\n\n    // Check file for locale exists\n    const localePath = path.join(basePath, locale, file);\n    const localeFileExists = await Utils.fileExists(localePath);\n\n    // If file for locale exists return file\n    if (localeFileExists) {\n      return { path: localePath, subdir: locale };\n    }\n\n    // Check file for language exists\n    const language = locale.split('-')[0];\n    const languagePath = path.join(basePath, language, file);\n    const languageFileExists = await Utils.fileExists(languagePath);\n\n    // If file for language exists return file\n    if (languageFileExists) {\n      return { path: languagePath, subdir: language };\n    }\n\n    // Return default file\n    return { path: defaultPath };\n  }\n\n  /**\n   * @function fileExists\n   * @description Checks whether a file exists.\n   * @param {String} path The file path.\n   * @returns {Promise<Boolean>} Is true if the file can be accessed, false otherwise.\n   */\n  static async fileExists(path) {\n    try {\n      await fs.access(path);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * @function isPath\n   * @description Evaluates whether a string is a file path (as opposed to a URL for example).\n   * @param {String} s The string to evaluate.\n   * @returns {Boolean} Returns true if the evaluated string is a path.\n   */\n  static isPath(s) {\n    return /(^\\/)|(^\\.\\/)|(^\\.\\.\\/)/.test(s);\n  }\n\n  /**\n   * Flattens an object and crates new keys with custom delimiters.\n   * @param {Object} obj The object to flatten.\n   * @param {String} [delimiter='.'] The delimiter of the newly generated keys.\n   * @param {Object} result\n   * @returns {Object} The flattened object.\n   **/\n  static flattenObject(obj, parentKey, delimiter = '.', result = {}) {\n    for (const key in obj) {\n      if (Object.prototype.hasOwnProperty.call(obj, key)) {\n        const newKey = parentKey ? parentKey + delimiter + key : key;\n\n        if (typeof obj[key] === 'object' && obj[key] !== null) {\n          this.flattenObject(obj[key], newKey, delimiter, result);\n        } else {\n          result[newKey] = obj[key];\n        }\n      }\n    }\n    return result;\n  }\n\n  /**\n   * Determines whether an object is a Promise.\n   * @param {any} object The object to validate.\n   * @returns {Boolean} Returns true if the object is a promise.\n   */\n  static isPromise(object) {\n    return object instanceof Promise;\n  }\n\n  /**\n   * Creates an object with all permutations of the original keys.\n   * For example, this definition:\n   * ```\n   * {\n   *   a: [true, false],\n   *   b: [1, 2],\n   *   c: ['x']\n   * }\n   * ```\n   * permutates to:\n   * ```\n   * [\n   *   { a: true, b: 1, c: 'x' },\n   *   { a: true, b: 2, c: 'x' },\n   *   { a: false, b: 1, c: 'x' },\n   *   { a: false, b: 2, c: 'x' }\n   * ]\n   * ```\n   * @param {Object} object The object to permutate.\n   * @param {Integer} [index=0] The current key index.\n   * @param {Object} [current={}] The current result entry being composed.\n   * @param {Array} [results=[]] The resulting array of permutations.\n   */\n  static getObjectKeyPermutations(object, index = 0, current = {}, results = []) {\n    const keys = Object.keys(object);\n    const key = keys[index];\n    const values = object[key];\n\n    for (const value of values) {\n      current[key] = value;\n      const nextIndex = index + 1;\n\n      if (nextIndex < keys.length) {\n        Utils.getObjectKeyPermutations(object, nextIndex, current, results);\n      } else {\n        const result = Object.assign({}, current);\n        results.push(result);\n      }\n    }\n    return results;\n  }\n\n  /**\n   * Validates parameters and throws if a parameter is invalid.\n   * Example parameter types syntax:\n   * ```\n   * {\n   *   parameterName: {\n   *      t: 'boolean',\n   *      v: isBoolean,\n   *      o: true\n   *   },\n   *   ...\n   * }\n   * ```\n   * @param {Object} params The parameters to validate.\n   * @param {Array<Object>} types The parameter types used for validation.\n   * @param {Object} types.t The parameter type; used for error message, not for validation.\n   * @param {Object} types.v The function to validate the parameter value.\n   * @param {Boolean} [types.o=false] Is true if the parameter is optional.\n   */\n  static validateParams(params, types) {\n    for (const key of Object.keys(params)) {\n      const type = types[key];\n      const isOptional = !!type.o;\n      const param = params[key];\n      if (!(isOptional && param == null) && !type.v(param)) {\n        throw `Invalid parameter ${key} must be of type ${type.t} but is ${typeof param}`;\n      }\n    }\n  }\n\n  /**\n   * Computes the relative date based on a string.\n   * @param {String} text The string to interpret the date from.\n   * @param {Date} now The date the string is comparing against.\n   * @returns {Object} The relative date object.\n   **/\n  static relativeTimeToDate(text, now = new Date()) {\n    text = text.toLowerCase();\n    let parts = text.split(' ');\n\n    // Filter out whitespace\n    parts = parts.filter(part => part !== '');\n\n    const future = parts[0] === 'in';\n    const past = parts[parts.length - 1] === 'ago';\n\n    if (!future && !past && text !== 'now') {\n      return {\n        status: 'error',\n        info: \"Time should either start with 'in' or end with 'ago'\",\n      };\n    }\n\n    if (future && past) {\n      return {\n        status: 'error',\n        info: \"Time cannot have both 'in' and 'ago'\",\n      };\n    }\n\n    // strip the 'ago' or 'in'\n    if (future) {\n      parts = parts.slice(1);\n    } else {\n      // past\n      parts = parts.slice(0, parts.length - 1);\n    }\n\n    if (parts.length % 2 !== 0 && text !== 'now') {\n      return {\n        status: 'error',\n        info: 'Invalid time string. Dangling unit or number.',\n      };\n    }\n\n    const pairs = [];\n    while (parts.length) {\n      pairs.push([parts.shift(), parts.shift()]);\n    }\n\n    let seconds = 0;\n    for (const [num, interval] of pairs) {\n      const val = Number(num);\n      if (!Number.isInteger(val)) {\n        return {\n          status: 'error',\n          info: `'${num}' is not an integer.`,\n        };\n      }\n\n      switch (interval) {\n        case 'yr':\n        case 'yrs':\n        case 'year':\n        case 'years':\n          seconds += val * 31536000; // 365 * 24 * 60 * 60\n          break;\n\n        case 'wk':\n        case 'wks':\n        case 'week':\n        case 'weeks':\n          seconds += val * 604800; // 7 * 24 * 60 * 60\n          break;\n\n        case 'd':\n        case 'day':\n        case 'days':\n          seconds += val * 86400; // 24 * 60 * 60\n          break;\n\n        case 'hr':\n        case 'hrs':\n        case 'hour':\n        case 'hours':\n          seconds += val * 3600; // 60 * 60\n          break;\n\n        case 'min':\n        case 'mins':\n        case 'minute':\n        case 'minutes':\n          seconds += val * 60;\n          break;\n\n        case 'sec':\n        case 'secs':\n        case 'second':\n        case 'seconds':\n          seconds += val;\n          break;\n\n        default:\n          return {\n            status: 'error',\n            info: `Invalid interval: '${interval}'`,\n          };\n      }\n    }\n\n    const milliseconds = seconds * 1000;\n    if (future) {\n      return {\n        status: 'success',\n        info: 'future',\n        result: new Date(now.valueOf() + milliseconds),\n      };\n    } else if (past) {\n      return {\n        status: 'success',\n        info: 'past',\n        result: new Date(now.valueOf() - milliseconds),\n      };\n    } else {\n      return {\n        status: 'success',\n        info: 'present',\n        result: new Date(now.valueOf()),\n      };\n    }\n  }\n\n  /**\n   * Deep-scans an object for a matching key/value definition.\n   * @param {Object} obj The object to scan.\n   * @param {String | undefined} key The key to match, or undefined if only the value should be matched.\n   * @param {any | undefined} value The value to match, or undefined if only the key should be matched.\n   * @returns {Boolean} True if a match was found, false otherwise.\n   */\n  static objectContainsKeyValue(obj, key, value) {\n    const isMatch = (a, b) => (typeof a === 'string' && new RegExp(b).test(a)) || a === b;\n    const isKeyMatch = k => isMatch(k, key);\n    const isValueMatch = v => isMatch(v, value);\n    for (const [k, v] of Object.entries(obj)) {\n      if (key !== undefined && value === undefined && isKeyMatch(k)) {\n        return true;\n      } else if (key === undefined && value !== undefined && isValueMatch(v)) {\n        return true;\n      } else if (key !== undefined && value !== undefined && isKeyMatch(k) && isValueMatch(v)) {\n        return true;\n      }\n      if (['[object Object]', '[object Array]'].includes(Object.prototype.toString.call(v))) {\n        return Utils.objectContainsKeyValue(v, key, value);\n      }\n    }\n    return false;\n  }\n\n  static checkProhibitedKeywords(config, data) {\n    if (config?.requestKeywordDenylist) {\n      // Scan request data for denied keywords\n      for (const keyword of config.requestKeywordDenylist) {\n        const match = Utils.objectContainsKeyValue(data, keyword.key, keyword.value);\n        if (match) {\n          throw `Prohibited keyword in request data: ${JSON.stringify(keyword)}.`;\n        }\n      }\n    }\n  }\n\n  /**\n   * Moves the nested keys of a specified key in an object to the root of the object.\n   *\n   * @param {Object} obj The object to modify.\n   * @param {String} key The key whose nested keys will be moved to root.\n   * @returns {Object} The modified object, or the original object if no modification happened.\n   * @example\n   * const obj = {\n   *   a: 1,\n   *   b: {\n   *     c: 2,\n   *     d: 3\n   *   },\n   *   e: 4\n   * };\n   * addNestedKeysToRoot(obj, 'b');\n   * console.log(obj);\n   * // Output: { a: 1, e: 4, c: 2, d: 3 }\n  */\n  static addNestedKeysToRoot(obj, key) {\n    if (obj[key] && typeof obj[key] === 'object') {\n      // Add nested keys to root\n      Object.assign(obj, { ...obj[key] });\n      // Delete original nested key\n      delete obj[key];\n    }\n    return obj;\n  }\n\n  /**\n   * Encodes a string to be used in a URL.\n   * @param {String} input The string to encode.\n   * @returns {String} The encoded string.\n   */\n  static encodeForUrl(input) {\n    return encodeURIComponent(input).replace(/[!'.()*]/g, char =>\n      '%' + char.charCodeAt(0).toString(16).toUpperCase()\n    );\n  }\n\n  /**\n   * Creates a JSON replacer function that handles Map, Set, and circular references.\n   * This replacer can be used with JSON.stringify to safely serialize complex objects.\n   *\n   * @returns {Function} A replacer function for JSON.stringify that:\n   * - Converts Map instances to plain objects\n   * - Converts Set instances to arrays\n   * - Replaces circular references with '[Circular]' marker\n   *\n   * @example\n   * const obj = { name: 'test', map: new Map([['key', 'value']]) };\n   * obj.self = obj; // circular reference\n   * JSON.stringify(obj, Utils.getCircularReplacer());\n   * // Output: {\"name\":\"test\",\"map\":{\"key\":\"value\"},\"self\":\"[Circular]\"}\n   */\n  static getCircularReplacer() {\n    const seen = new WeakSet();\n    return (key, value) => {\n      if (value instanceof Map) {\n        return Object.fromEntries(value);\n      }\n      if (value instanceof Set) {\n        return Array.from(value);\n      }\n      if (typeof value === 'object' && value !== null) {\n        if (seen.has(value)) {\n          return '[Circular]';\n        }\n        seen.add(value);\n      }\n      return value;\n    };\n  }\n\n  /**\n   * Gets a nested property value from an object using dot notation.\n   * @param {Object} obj The object to get the property from.\n   * @param {String} path The property path in dot notation, e.g. 'databaseOptions.allowPublicExplain'.\n   * @returns {any} The property value or undefined if not found.\n   * @example\n   * const obj = { database: { options: { enabled: true } } };\n   * Utils.getNestedProperty(obj, 'database.options.enabled');\n   * // Output: true\n   */\n  static getNestedProperty(obj, path) {\n    if (!obj || !path) {\n      return undefined;\n    }\n    const keys = path.split('.');\n    let current = obj;\n    for (const key of keys) {\n      if (current == null || typeof current !== 'object') {\n        return undefined;\n      }\n      current = current[key];\n    }\n    return current;\n  }\n\n  /**\n   * Parses a human-readable size string into a byte count.\n   * @param {number | string} size - A number (floored to an integer), a numeric string\n   *   (treated as bytes), or a string with a unit suffix: `b`, `kb`, `mb`, `gb`\n   *   (case-insensitive). Examples: `'20mb'`, `'512kb'`, `'1.5gb'`, `1048576`.\n   * @returns {number} The size in bytes, floored to the nearest integer.\n   * @throws {Error} If the string does not match the expected format.\n   */\n  static parseSizeToBytes(size) {\n    if (typeof size === 'number') {\n      if (!Number.isFinite(size) || size < 0) {\n        throw new Error(`Invalid size value: ${size}`);\n      }\n      return Math.floor(size);\n    }\n    const str = String(size).trim().toLowerCase();\n    const match = str.match(/^(\\d+(?:\\.\\d+)?)\\s*(b|kb|mb|gb)?$/);\n    if (!match) {\n      throw new Error(`Invalid size value: ${size}`);\n    }\n    const num = parseFloat(match[1]);\n    const unit = match[2];\n    switch (unit) {\n      case 'kb':\n        return Math.floor(num * 1024);\n      case 'mb':\n        return Math.floor(num * 1024 * 1024);\n      case 'gb':\n        return Math.floor(num * 1024 * 1024 * 1024);\n      default:\n        return Math.floor(num);\n    }\n  }\n}\n\nmodule.exports = Utils;\n"],"mappings":";;AAAA;AACA;AACA;AACA;AACA;;AAEA,MAAMA,IAAI,GAAGC,OAAO,CAAC,MAAM,CAAC;AAC5B,MAAMC,EAAE,GAAGD,OAAO,CAAC,IAAI,CAAC,CAACE,QAAQ;;AAEjC;AACA;AACA;AACA,MAAMC,KAAK,CAAC;EACV;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACE,aAAaC,gBAAgBA,CAACC,WAAW,EAAEC,MAAM,EAAE;IACjD;IACA,MAAMC,IAAI,GAAGR,IAAI,CAACS,QAAQ,CAACH,WAAW,CAAC;IACvC,MAAMI,QAAQ,GAAGV,IAAI,CAACW,OAAO,CAACL,WAAW,CAAC;;IAE1C;IACA,IAAI,CAACC,MAAM,EAAE;MACX,OAAO;QAAEP,IAAI,EAAEM;MAAY,CAAC;IAC9B;;IAEA;IACA,MAAMM,UAAU,GAAGZ,IAAI,CAACa,IAAI,CAACH,QAAQ,EAAEH,MAAM,EAAEC,IAAI,CAAC;IACpD,MAAMM,gBAAgB,GAAG,MAAMV,KAAK,CAACW,UAAU,CAACH,UAAU,CAAC;;IAE3D;IACA,IAAIE,gBAAgB,EAAE;MACpB,OAAO;QAAEd,IAAI,EAAEY,UAAU;QAAEI,MAAM,EAAET;MAAO,CAAC;IAC7C;;IAEA;IACA,MAAMU,QAAQ,GAAGV,MAAM,CAACW,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;IACrC,MAAMC,YAAY,GAAGnB,IAAI,CAACa,IAAI,CAACH,QAAQ,EAAEO,QAAQ,EAAET,IAAI,CAAC;IACxD,MAAMY,kBAAkB,GAAG,MAAMhB,KAAK,CAACW,UAAU,CAACI,YAAY,CAAC;;IAE/D;IACA,IAAIC,kBAAkB,EAAE;MACtB,OAAO;QAAEpB,IAAI,EAAEmB,YAAY;QAAEH,MAAM,EAAEC;MAAS,CAAC;IACjD;;IAEA;IACA,OAAO;MAAEjB,IAAI,EAAEM;IAAY,CAAC;EAC9B;;EAEA;AACF;AACA;AACA;AACA;AACA;EACE,aAAaS,UAAUA,CAACf,IAAI,EAAE;IAC5B,IAAI;MACF,MAAME,EAAE,CAACmB,MAAM,CAACrB,IAAI,CAAC;MACrB,OAAO,IAAI;IACb,CAAC,CAAC,MAAM;MACN,OAAO,KAAK;IACd;EACF;;EAEA;AACF;AACA;AACA;AACA;AACA;EACE,OAAOsB,MAAMA,CAACC,CAAC,EAAE;IACf,OAAO,yBAAyB,CAACC,IAAI,CAACD,CAAC,CAAC;EAC1C;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;EACE,OAAOE,aAAaA,CAACC,GAAG,EAAEC,SAAS,EAAEC,SAAS,GAAG,GAAG,EAAEC,MAAM,GAAG,CAAC,CAAC,EAAE;IACjE,KAAK,MAAMC,GAAG,IAAIJ,GAAG,EAAE;MACrB,IAAIK,MAAM,CAACC,SAAS,CAACC,cAAc,CAACC,IAAI,CAACR,GAAG,EAAEI,GAAG,CAAC,EAAE;QAClD,MAAMK,MAAM,GAAGR,SAAS,GAAGA,SAAS,GAAGC,SAAS,GAAGE,GAAG,GAAGA,GAAG;QAE5D,IAAI,OAAOJ,GAAG,CAACI,GAAG,CAAC,KAAK,QAAQ,IAAIJ,GAAG,CAACI,GAAG,CAAC,KAAK,IAAI,EAAE;UACrD,IAAI,CAACL,aAAa,CAACC,GAAG,CAACI,GAAG,CAAC,EAAEK,MAAM,EAAEP,SAAS,EAAEC,MAAM,CAAC;QACzD,CAAC,MAAM;UACLA,MAAM,CAACM,MAAM,CAAC,GAAGT,GAAG,CAACI,GAAG,CAAC;QAC3B;MACF;IACF;IACA,OAAOD,MAAM;EACf;;EAEA;AACF;AACA;AACA;AACA;EACE,OAAOO,SAASA,CAACC,MAAM,EAAE;IACvB,OAAOA,MAAM,YAAYC,OAAO;EAClC;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACE,OAAOC,wBAAwBA,CAACF,MAAM,EAAEG,KAAK,GAAG,CAAC,EAAEC,OAAO,GAAG,CAAC,CAAC,EAAEC,OAAO,GAAG,EAAE,EAAE;IAC7E,MAAMC,IAAI,GAAGZ,MAAM,CAACY,IAAI,CAACN,MAAM,CAAC;IAChC,MAAMP,GAAG,GAAGa,IAAI,CAACH,KAAK,CAAC;IACvB,MAAMI,MAAM,GAAGP,MAAM,CAACP,GAAG,CAAC;IAE1B,KAAK,MAAMe,KAAK,IAAID,MAAM,EAAE;MAC1BH,OAAO,CAACX,GAAG,CAAC,GAAGe,KAAK;MACpB,MAAMC,SAAS,GAAGN,KAAK,GAAG,CAAC;MAE3B,IAAIM,SAAS,GAAGH,IAAI,CAACI,MAAM,EAAE;QAC3B3C,KAAK,CAACmC,wBAAwB,CAACF,MAAM,EAAES,SAAS,EAAEL,OAAO,EAAEC,OAAO,CAAC;MACrE,CAAC,MAAM;QACL,MAAMb,MAAM,GAAGE,MAAM,CAACiB,MAAM,CAAC,CAAC,CAAC,EAAEP,OAAO,CAAC;QACzCC,OAAO,CAACO,IAAI,CAACpB,MAAM,CAAC;MACtB;IACF;IACA,OAAOa,OAAO;EAChB;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACE,OAAOQ,cAAcA,CAACC,MAAM,EAAEC,KAAK,EAAE;IACnC,KAAK,MAAMtB,GAAG,IAAIC,MAAM,CAACY,IAAI,CAACQ,MAAM,CAAC,EAAE;MACrC,MAAME,IAAI,GAAGD,KAAK,CAACtB,GAAG,CAAC;MACvB,MAAMwB,UAAU,GAAG,CAAC,CAACD,IAAI,CAACE,CAAC;MAC3B,MAAMC,KAAK,GAAGL,MAAM,CAACrB,GAAG,CAAC;MACzB,IAAI,EAAEwB,UAAU,IAAIE,KAAK,IAAI,IAAI,CAAC,IAAI,CAACH,IAAI,CAACI,CAAC,CAACD,KAAK,CAAC,EAAE;QACpD,MAAM,qBAAqB1B,GAAG,oBAAoBuB,IAAI,CAACK,CAAC,WAAW,OAAOF,KAAK,EAAE;MACnF;IACF;EACF;;EAEA;AACF;AACA;AACA;AACA;AACA;EACE,OAAOG,kBAAkBA,CAACC,IAAI,EAAEC,GAAG,GAAG,IAAIC,IAAI,CAAC,CAAC,EAAE;IAChDF,IAAI,GAAGA,IAAI,CAACG,WAAW,CAAC,CAAC;IACzB,IAAIC,KAAK,GAAGJ,IAAI,CAAC1C,KAAK,CAAC,GAAG,CAAC;;IAE3B;IACA8C,KAAK,GAAGA,KAAK,CAACC,MAAM,CAACC,IAAI,IAAIA,IAAI,KAAK,EAAE,CAAC;IAEzC,MAAMC,MAAM,GAAGH,KAAK,CAAC,CAAC,CAAC,KAAK,IAAI;IAChC,MAAMI,IAAI,GAAGJ,KAAK,CAACA,KAAK,CAACjB,MAAM,GAAG,CAAC,CAAC,KAAK,KAAK;IAE9C,IAAI,CAACoB,MAAM,IAAI,CAACC,IAAI,IAAIR,IAAI,KAAK,KAAK,EAAE;MACtC,OAAO;QACLS,MAAM,EAAE,OAAO;QACfC,IAAI,EAAE;MACR,CAAC;IACH;IAEA,IAAIH,MAAM,IAAIC,IAAI,EAAE;MAClB,OAAO;QACLC,MAAM,EAAE,OAAO;QACfC,IAAI,EAAE;MACR,CAAC;IACH;;IAEA;IACA,IAAIH,MAAM,EAAE;MACVH,KAAK,GAAGA,KAAK,CAACO,KAAK,CAAC,CAAC,CAAC;IACxB,CAAC,MAAM;MACL;MACAP,KAAK,GAAGA,KAAK,CAACO,KAAK,CAAC,CAAC,EAAEP,KAAK,CAACjB,MAAM,GAAG,CAAC,CAAC;IAC1C;IAEA,IAAIiB,KAAK,CAACjB,MAAM,GAAG,CAAC,KAAK,CAAC,IAAIa,IAAI,KAAK,KAAK,EAAE;MAC5C,OAAO;QACLS,MAAM,EAAE,OAAO;QACfC,IAAI,EAAE;MACR,CAAC;IACH;IAEA,MAAME,KAAK,GAAG,EAAE;IAChB,OAAOR,KAAK,CAACjB,MAAM,EAAE;MACnByB,KAAK,CAACvB,IAAI,CAAC,CAACe,KAAK,CAACS,KAAK,CAAC,CAAC,EAAET,KAAK,CAACS,KAAK,CAAC,CAAC,CAAC,CAAC;IAC5C;IAEA,IAAIC,OAAO,GAAG,CAAC;IACf,KAAK,MAAM,CAACC,GAAG,EAAEC,QAAQ,CAAC,IAAIJ,KAAK,EAAE;MACnC,MAAMK,GAAG,GAAGC,MAAM,CAACH,GAAG,CAAC;MACvB,IAAI,CAACG,MAAM,CAACC,SAAS,CAACF,GAAG,CAAC,EAAE;QAC1B,OAAO;UACLR,MAAM,EAAE,OAAO;UACfC,IAAI,EAAE,IAAIK,GAAG;QACf,CAAC;MACH;MAEA,QAAQC,QAAQ;QACd,KAAK,IAAI;QACT,KAAK,KAAK;QACV,KAAK,MAAM;QACX,KAAK,OAAO;UACVF,OAAO,IAAIG,GAAG,GAAG,QAAQ,CAAC,CAAC;UAC3B;QAEF,KAAK,IAAI;QACT,KAAK,KAAK;QACV,KAAK,MAAM;QACX,KAAK,OAAO;UACVH,OAAO,IAAIG,GAAG,GAAG,MAAM,CAAC,CAAC;UACzB;QAEF,KAAK,GAAG;QACR,KAAK,KAAK;QACV,KAAK,MAAM;UACTH,OAAO,IAAIG,GAAG,GAAG,KAAK,CAAC,CAAC;UACxB;QAEF,KAAK,IAAI;QACT,KAAK,KAAK;QACV,KAAK,MAAM;QACX,KAAK,OAAO;UACVH,OAAO,IAAIG,GAAG,GAAG,IAAI,CAAC,CAAC;UACvB;QAEF,KAAK,KAAK;QACV,KAAK,MAAM;QACX,KAAK,QAAQ;QACb,KAAK,SAAS;UACZH,OAAO,IAAIG,GAAG,GAAG,EAAE;UACnB;QAEF,KAAK,KAAK;QACV,KAAK,MAAM;QACX,KAAK,QAAQ;QACb,KAAK,SAAS;UACZH,OAAO,IAAIG,GAAG;UACd;QAEF;UACE,OAAO;YACLR,MAAM,EAAE,OAAO;YACfC,IAAI,EAAE,sBAAsBM,QAAQ;UACtC,CAAC;MACL;IACF;IAEA,MAAMI,YAAY,GAAGN,OAAO,GAAG,IAAI;IACnC,IAAIP,MAAM,EAAE;MACV,OAAO;QACLE,MAAM,EAAE,SAAS;QACjBC,IAAI,EAAE,QAAQ;QACdzC,MAAM,EAAE,IAAIiC,IAAI,CAACD,GAAG,CAACoB,OAAO,CAAC,CAAC,GAAGD,YAAY;MAC/C,CAAC;IACH,CAAC,MAAM,IAAIZ,IAAI,EAAE;MACf,OAAO;QACLC,MAAM,EAAE,SAAS;QACjBC,IAAI,EAAE,MAAM;QACZzC,MAAM,EAAE,IAAIiC,IAAI,CAACD,GAAG,CAACoB,OAAO,CAAC,CAAC,GAAGD,YAAY;MAC/C,CAAC;IACH,CAAC,MAAM;MACL,OAAO;QACLX,MAAM,EAAE,SAAS;QACjBC,IAAI,EAAE,SAAS;QACfzC,MAAM,EAAE,IAAIiC,IAAI,CAACD,GAAG,CAACoB,OAAO,CAAC,CAAC;MAChC,CAAC;IACH;EACF;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;EACE,OAAOC,sBAAsBA,CAACxD,GAAG,EAAEI,GAAG,EAAEe,KAAK,EAAE;IAC7C,MAAMsC,OAAO,GAAGA,CAACC,CAAC,EAAEC,CAAC,KAAM,OAAOD,CAAC,KAAK,QAAQ,IAAI,IAAIE,MAAM,CAACD,CAAC,CAAC,CAAC7D,IAAI,CAAC4D,CAAC,CAAC,IAAKA,CAAC,KAAKC,CAAC;IACrF,MAAME,UAAU,GAAGC,CAAC,IAAIL,OAAO,CAACK,CAAC,EAAE1D,GAAG,CAAC;IACvC,MAAM2D,YAAY,GAAGhC,CAAC,IAAI0B,OAAO,CAAC1B,CAAC,EAAEZ,KAAK,CAAC;IAC3C,KAAK,MAAM,CAAC2C,CAAC,EAAE/B,CAAC,CAAC,IAAI1B,MAAM,CAAC2D,OAAO,CAAChE,GAAG,CAAC,EAAE;MACxC,IAAII,GAAG,KAAK6D,SAAS,IAAI9C,KAAK,KAAK8C,SAAS,IAAIJ,UAAU,CAACC,CAAC,CAAC,EAAE;QAC7D,OAAO,IAAI;MACb,CAAC,MAAM,IAAI1D,GAAG,KAAK6D,SAAS,IAAI9C,KAAK,KAAK8C,SAAS,IAAIF,YAAY,CAAChC,CAAC,CAAC,EAAE;QACtE,OAAO,IAAI;MACb,CAAC,MAAM,IAAI3B,GAAG,KAAK6D,SAAS,IAAI9C,KAAK,KAAK8C,SAAS,IAAIJ,UAAU,CAACC,CAAC,CAAC,IAAIC,YAAY,CAAChC,CAAC,CAAC,EAAE;QACvF,OAAO,IAAI;MACb;MACA,IAAI,CAAC,iBAAiB,EAAE,gBAAgB,CAAC,CAACmC,QAAQ,CAAC7D,MAAM,CAACC,SAAS,CAAC6D,QAAQ,CAAC3D,IAAI,CAACuB,CAAC,CAAC,CAAC,EAAE;QACrF,OAAOrD,KAAK,CAAC8E,sBAAsB,CAACzB,CAAC,EAAE3B,GAAG,EAAEe,KAAK,CAAC;MACpD;IACF;IACA,OAAO,KAAK;EACd;EAEA,OAAOiD,uBAAuBA,CAACC,MAAM,EAAEC,IAAI,EAAE;IAC3C,IAAID,MAAM,EAAEE,sBAAsB,EAAE;MAClC;MACA,KAAK,MAAMC,OAAO,IAAIH,MAAM,CAACE,sBAAsB,EAAE;QACnD,MAAME,KAAK,GAAG/F,KAAK,CAAC8E,sBAAsB,CAACc,IAAI,EAAEE,OAAO,CAACpE,GAAG,EAAEoE,OAAO,CAACrD,KAAK,CAAC;QAC5E,IAAIsD,KAAK,EAAE;UACT,MAAM,uCAAuCC,IAAI,CAACC,SAAS,CAACH,OAAO,CAAC,GAAG;QACzE;MACF;IACF;EACF;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACE,OAAOI,mBAAmBA,CAAC5E,GAAG,EAAEI,GAAG,EAAE;IACnC,IAAIJ,GAAG,CAACI,GAAG,CAAC,IAAI,OAAOJ,GAAG,CAACI,GAAG,CAAC,KAAK,QAAQ,EAAE;MAC5C;MACAC,MAAM,CAACiB,MAAM,CAACtB,GAAG,EAAE;QAAE,GAAGA,GAAG,CAACI,GAAG;MAAE,CAAC,CAAC;MACnC;MACA,OAAOJ,GAAG,CAACI,GAAG,CAAC;IACjB;IACA,OAAOJ,GAAG;EACZ;;EAEA;AACF;AACA;AACA;AACA;EACE,OAAO6E,YAAYA,CAACC,KAAK,EAAE;IACzB,OAAOC,kBAAkB,CAACD,KAAK,CAAC,CAACE,OAAO,CAAC,WAAW,EAAEC,IAAI,IACxD,GAAG,GAAGA,IAAI,CAACC,UAAU,CAAC,CAAC,CAAC,CAACf,QAAQ,CAAC,EAAE,CAAC,CAACgB,WAAW,CAAC,CACpD,CAAC;EACH;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACE,OAAOC,mBAAmBA,CAAA,EAAG;IAC3B,MAAMC,IAAI,GAAG,IAAIC,OAAO,CAAC,CAAC;IAC1B,OAAO,CAAClF,GAAG,EAAEe,KAAK,KAAK;MACrB,IAAIA,KAAK,YAAYoE,GAAG,EAAE;QACxB,OAAOlF,MAAM,CAACmF,WAAW,CAACrE,KAAK,CAAC;MAClC;MACA,IAAIA,KAAK,YAAYsE,GAAG,EAAE;QACxB,OAAOC,KAAK,CAACC,IAAI,CAACxE,KAAK,CAAC;MAC1B;MACA,IAAI,OAAOA,KAAK,KAAK,QAAQ,IAAIA,KAAK,KAAK,IAAI,EAAE;QAC/C,IAAIkE,IAAI,CAACO,GAAG,CAACzE,KAAK,CAAC,EAAE;UACnB,OAAO,YAAY;QACrB;QACAkE,IAAI,CAACQ,GAAG,CAAC1E,KAAK,CAAC;MACjB;MACA,OAAOA,KAAK;IACd,CAAC;EACH;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACE,OAAO2E,iBAAiBA,CAAC9F,GAAG,EAAE1B,IAAI,EAAE;IAClC,IAAI,CAAC0B,GAAG,IAAI,CAAC1B,IAAI,EAAE;MACjB,OAAO2F,SAAS;IAClB;IACA,MAAMhD,IAAI,GAAG3C,IAAI,CAACkB,KAAK,CAAC,GAAG,CAAC;IAC5B,IAAIuB,OAAO,GAAGf,GAAG;IACjB,KAAK,MAAMI,GAAG,IAAIa,IAAI,EAAE;MACtB,IAAIF,OAAO,IAAI,IAAI,IAAI,OAAOA,OAAO,KAAK,QAAQ,EAAE;QAClD,OAAOkD,SAAS;MAClB;MACAlD,OAAO,GAAGA,OAAO,CAACX,GAAG,CAAC;IACxB;IACA,OAAOW,OAAO;EAChB;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;EACE,OAAOgF,gBAAgBA,CAACC,IAAI,EAAE;IAC5B,IAAI,OAAOA,IAAI,KAAK,QAAQ,EAAE;MAC5B,IAAI,CAAC5C,MAAM,CAAC6C,QAAQ,CAACD,IAAI,CAAC,IAAIA,IAAI,GAAG,CAAC,EAAE;QACtC,MAAM,IAAIE,KAAK,CAAC,uBAAuBF,IAAI,EAAE,CAAC;MAChD;MACA,OAAOG,IAAI,CAACC,KAAK,CAACJ,IAAI,CAAC;IACzB;IACA,MAAMK,GAAG,GAAGC,MAAM,CAACN,IAAI,CAAC,CAACO,IAAI,CAAC,CAAC,CAAClE,WAAW,CAAC,CAAC;IAC7C,MAAMoC,KAAK,GAAG4B,GAAG,CAAC5B,KAAK,CAAC,mCAAmC,CAAC;IAC5D,IAAI,CAACA,KAAK,EAAE;MACV,MAAM,IAAIyB,KAAK,CAAC,uBAAuBF,IAAI,EAAE,CAAC;IAChD;IACA,MAAM/C,GAAG,GAAGuD,UAAU,CAAC/B,KAAK,CAAC,CAAC,CAAC,CAAC;IAChC,MAAMgC,IAAI,GAAGhC,KAAK,CAAC,CAAC,CAAC;IACrB,QAAQgC,IAAI;MACV,KAAK,IAAI;QACP,OAAON,IAAI,CAACC,KAAK,CAACnD,GAAG,GAAG,IAAI,CAAC;MAC/B,KAAK,IAAI;QACP,OAAOkD,IAAI,CAACC,KAAK,CAACnD,GAAG,GAAG,IAAI,GAAG,IAAI,CAAC;MACtC,KAAK,IAAI;QACP,OAAOkD,IAAI,CAACC,KAAK,CAACnD,GAAG,GAAG,IAAI,GAAG,IAAI,GAAG,IAAI,CAAC;MAC7C;QACE,OAAOkD,IAAI,CAACC,KAAK,CAACnD,GAAG,CAAC;IAC1B;EACF;AACF;AAEAyD,MAAM,CAACC,OAAO,GAAGjI,KAAK","ignoreList":[]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "parse-server",
3
- "version": "9.3.0-alpha.8",
3
+ "version": "9.3.0-alpha.9",
4
4
  "description": "An express module providing a Parse-compatible API server",
5
5
  "main": "lib/index.js",
6
6
  "repository": {