parse-server 8.1.0-alpha.1 → 8.1.0-alpha.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/lib/LiveQuery/ParseLiveQueryServer.js +13 -3
- package/lib/ParseServer.js +48 -32
- package/lib/Routers/FilesRouter.js +55 -17
- package/lib/TestUtils.js +39 -1
- package/lib/triggers.js +4 -1
- package/package.json +8 -8
|
@@ -68,26 +68,64 @@ class FilesRouter {
|
|
|
68
68
|
});
|
|
69
69
|
return;
|
|
70
70
|
}
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
71
|
+
let filename = req.params.filename;
|
|
72
|
+
try {
|
|
73
|
+
const filesController = config.filesController;
|
|
74
|
+
const mime = (await import('mime')).default;
|
|
75
|
+
let contentType = mime.getType(filename);
|
|
76
|
+
let file = new _node.default.File(filename, {
|
|
77
|
+
base64: ''
|
|
78
|
+
}, contentType);
|
|
79
|
+
const triggerResult = await triggers.maybeRunFileTrigger(triggers.Types.beforeFind, {
|
|
80
|
+
file
|
|
81
|
+
}, config, req.auth);
|
|
82
|
+
if (triggerResult?.file?._name) {
|
|
83
|
+
filename = triggerResult?.file?._name;
|
|
84
|
+
contentType = mime.getType(filename);
|
|
85
|
+
}
|
|
86
|
+
if (isFileStreamable(req, filesController)) {
|
|
87
|
+
filesController.handleFileStream(config, filename, req, res, contentType).catch(() => {
|
|
88
|
+
res.status(404);
|
|
89
|
+
res.set('Content-Type', 'text/plain');
|
|
90
|
+
res.end('File not found.');
|
|
91
|
+
});
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
let data = await filesController.getFileData(config, filename).catch(() => {
|
|
77
95
|
res.status(404);
|
|
78
96
|
res.set('Content-Type', 'text/plain');
|
|
79
97
|
res.end('File not found.');
|
|
80
98
|
});
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
99
|
+
if (!data) {
|
|
100
|
+
return;
|
|
101
|
+
}
|
|
102
|
+
file = new _node.default.File(filename, {
|
|
103
|
+
base64: data.toString('base64')
|
|
104
|
+
}, contentType);
|
|
105
|
+
const afterFind = await triggers.maybeRunFileTrigger(triggers.Types.afterFind, {
|
|
106
|
+
file,
|
|
107
|
+
forceDownload: false
|
|
108
|
+
}, config, req.auth);
|
|
109
|
+
if (afterFind?.file) {
|
|
110
|
+
contentType = mime.getType(afterFind.file._name);
|
|
111
|
+
data = Buffer.from(afterFind.file._data, 'base64');
|
|
112
|
+
}
|
|
113
|
+
res.status(200);
|
|
114
|
+
res.set('Content-Type', contentType);
|
|
115
|
+
res.set('Content-Length', data.length);
|
|
116
|
+
if (afterFind.forceDownload) {
|
|
117
|
+
res.set('Content-Disposition', `attachment;filename=${afterFind.file._name}`);
|
|
118
|
+
}
|
|
119
|
+
res.end(data);
|
|
120
|
+
} catch (e) {
|
|
121
|
+
const err = triggers.resolveError(e, {
|
|
122
|
+
code: _node.default.Error.SCRIPT_FAILED,
|
|
123
|
+
message: `Could not find file: ${filename}.`
|
|
124
|
+
});
|
|
125
|
+
res.status(403);
|
|
126
|
+
res.json({
|
|
127
|
+
code: err.code,
|
|
128
|
+
error: err.message
|
|
91
129
|
});
|
|
92
130
|
}
|
|
93
131
|
}
|
|
@@ -286,4 +324,4 @@ function isFileStreamable(req, filesController) {
|
|
|
286
324
|
const end = Number(range[1]);
|
|
287
325
|
return (!isNaN(start) || !isNaN(end)) && typeof filesController.adapter.handleFileStream === 'function';
|
|
288
326
|
}
|
|
289
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_express","_interopRequireDefault","require","Middlewares","_interopRequireWildcard","_node","_Config","_logger","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","triggers","http","Utils","downloadFileFromURI","uri","Promise","res","rej","response","setDefaultEncoding","body","headers","on","data","message","addFileDataIfNeeded","file","_source","format","base64","_previousSave","_data","_requestTask","FilesRouter","expressRouter","maxUploadSize","router","express","Router","getHandler","metadataHandler","post","req","next","Parse","Error","INVALID_FILE_NAME","raw","type","limit","handleParseHeaders","handleParseSession","createHandler","delete","enforceMasterKeyAccess","deleteHandler","config","Config","params","appId","status","err","OPERATION_FORBIDDEN","json","code","error","filesController","filename","mime","contentType","getType","isFileStreamable","handleFileStream","catch","end","getFileData","then","length","user","auth","isMaster","isLinked","AnonymousUtils","fileUpload","enableForAnonymousUser","FILE_SAVE_ERROR","enableForAuthenticatedUser","enableForPublic","validateFilename","fileExtensions","isValidExtension","extension","some","ext","regex","RegExp","test","includes","substring","lastIndexOf","split","join","toString","File","metadata","tags","fileData","checkProhibitedKeywords","INVALID_KEY_NAME","setTags","setMetadata","fileSize","Buffer","byteLength","fileObject","triggerResult","maybeRunFileTrigger","Types","beforeSave","saveResult","url","name","_name","bufferData","from","fileOptions","_metadata","fileTags","keys","_tags","assign","createFileResult","createFile","_url","resolve","afterSave","logger","resolveError","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 http = require('http');\nconst Utils = require('../Utils');\n\nconst downloadFileFromURI = uri => {\n  return new Promise((res, rej) => {\n    http\n      .get(uri, response => {\n        response.setDefaultEncoding('base64');\n        let body = `data:${response.headers['content-type']};base64,`;\n        response.on('data', data => (body += data));\n        response.on('end', () => res(body));\n      })\n      .on('error', e => {\n        rej(`Error downloading file from ${uri}: ${e.message}`);\n      });\n  });\n};\n\nconst addFileDataIfNeeded = async file => {\n  if (file._source.format === 'uri') {\n    const base64 = await downloadFileFromURI(file._source.uri);\n    file._previousSave = file;\n    file._data = base64;\n    file._requestTask = null;\n  }\n  return file;\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      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      const err = new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, 'Invalid application ID.');\n      res.json({ code: err.code, error: err.message });\n      return;\n    }\n    const filesController = config.filesController;\n    const filename = req.params.filename;\n    const mime = (await import('mime')).default;\n    const contentType = mime.getType(filename);\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    } else {\n      filesController\n        .getFileData(config, filename)\n        .then(data => {\n          res.status(200);\n          res.set('Content-Type', contentType);\n          res.set('Content-Length', data.length);\n          res.end(data);\n        })\n        .catch(() => {\n          res.status(404);\n          res.set('Content-Type', 'text/plain');\n          res.end('File not found.');\n        });\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        // if the ParseFile returned is type uri, download the file before saving it\n        await addFileDataIfNeeded(fileObject.file);\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 (e) {\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":[]}
|
|
327
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_express","_interopRequireDefault","require","Middlewares","_interopRequireWildcard","_node","_Config","_logger","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","triggers","http","Utils","downloadFileFromURI","uri","Promise","res","rej","response","setDefaultEncoding","body","headers","on","data","message","addFileDataIfNeeded","file","_source","format","base64","_previousSave","_data","_requestTask","FilesRouter","expressRouter","maxUploadSize","router","express","Router","getHandler","metadataHandler","post","req","next","Parse","Error","INVALID_FILE_NAME","raw","type","limit","handleParseHeaders","handleParseSession","createHandler","delete","enforceMasterKeyAccess","deleteHandler","config","Config","params","appId","status","err","OPERATION_FORBIDDEN","json","code","error","filename","filesController","mime","contentType","getType","File","triggerResult","maybeRunFileTrigger","Types","beforeFind","auth","_name","isFileStreamable","handleFileStream","catch","end","getFileData","toString","afterFind","forceDownload","Buffer","from","length","resolveError","SCRIPT_FAILED","user","isMaster","isLinked","AnonymousUtils","fileUpload","enableForAnonymousUser","FILE_SAVE_ERROR","enableForAuthenticatedUser","enableForPublic","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","_url","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 http = require('http');\nconst Utils = require('../Utils');\n\nconst downloadFileFromURI = uri => {\n  return new Promise((res, rej) => {\n    http\n      .get(uri, response => {\n        response.setDefaultEncoding('base64');\n        let body = `data:${response.headers['content-type']};base64,`;\n        response.on('data', data => (body += data));\n        response.on('end', () => res(body));\n      })\n      .on('error', e => {\n        rej(`Error downloading file from ${uri}: ${e.message}`);\n      });\n  });\n};\n\nconst addFileDataIfNeeded = async file => {\n  if (file._source.format === 'uri') {\n    const base64 = await downloadFileFromURI(file._source.uri);\n    file._previousSave = file;\n    file._data = base64;\n    file._requestTask = null;\n  }\n  return file;\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      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      const err = new Parse.Error(Parse.Error.OPERATION_FORBIDDEN, 'Invalid application ID.');\n      res.json({ code: err.code, error: err.message });\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        // if the ParseFile returned is type uri, download the file before saving it\n        await addFileDataIfNeeded(fileObject.file);\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 (e) {\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,SAAAM,yBAAAC,CAAA,6BAAAC,OAAA,mBAAAC,CAAA,OAAAD,OAAA,IAAAE,CAAA,OAAAF,OAAA,YAAAF,wBAAA,YAAAA,CAAAC,CAAA,WAAAA,CAAA,GAAAG,CAAA,GAAAD,CAAA,KAAAF,CAAA;AAAA,SAAAL,wBAAAK,CAAA,EAAAE,CAAA,SAAAA,CAAA,IAAAF,CAAA,IAAAA,CAAA,CAAAI,UAAA,SAAAJ,CAAA,eAAAA,CAAA,uBAAAA,CAAA,yBAAAA,CAAA,WAAAK,OAAA,EAAAL,CAAA,QAAAG,CAAA,GAAAJ,wBAAA,CAAAG,CAAA,OAAAC,CAAA,IAAAA,CAAA,CAAAG,GAAA,CAAAN,CAAA,UAAAG,CAAA,CAAAI,GAAA,CAAAP,CAAA,OAAAQ,CAAA,KAAAC,SAAA,UAAAC,CAAA,GAAAC,MAAA,CAAAC,cAAA,IAAAD,MAAA,CAAAE,wBAAA,WAAAC,CAAA,IAAAd,CAAA,oBAAAc,CAAA,OAAAC,cAAA,CAAAC,IAAA,CAAAhB,CAAA,EAAAc,CAAA,SAAAG,CAAA,GAAAP,CAAA,GAAAC,MAAA,CAAAE,wBAAA,CAAAb,CAAA,EAAAc,CAAA,UAAAG,CAAA,KAAAA,CAAA,CAAAV,GAAA,IAAAU,CAAA,CAAAC,GAAA,IAAAP,MAAA,CAAAC,cAAA,CAAAJ,CAAA,EAAAM,CAAA,EAAAG,CAAA,IAAAT,CAAA,CAAAM,CAAA,IAAAd,CAAA,CAAAc,CAAA,YAAAN,CAAA,CAAAH,OAAA,GAAAL,CAAA,EAAAG,CAAA,IAAAA,CAAA,CAAAe,GAAA,CAAAlB,CAAA,EAAAQ,CAAA,GAAAA,CAAA;AAAA,SAAAhB,uBAAAQ,CAAA,WAAAA,CAAA,IAAAA,CAAA,CAAAI,UAAA,GAAAJ,CAAA,KAAAK,OAAA,EAAAL,CAAA;AAC/B,MAAMmB,QAAQ,GAAG1B,OAAO,CAAC,aAAa,CAAC;AACvC,MAAM2B,IAAI,GAAG3B,OAAO,CAAC,MAAM,CAAC;AAC5B,MAAM4B,KAAK,GAAG5B,OAAO,CAAC,UAAU,CAAC;AAEjC,MAAM6B,mBAAmB,GAAGC,GAAG,IAAI;EACjC,OAAO,IAAIC,OAAO,CAAC,CAACC,GAAG,EAAEC,GAAG,KAAK;IAC/BN,IAAI,CACDb,GAAG,CAACgB,GAAG,EAAEI,QAAQ,IAAI;MACpBA,QAAQ,CAACC,kBAAkB,CAAC,QAAQ,CAAC;MACrC,IAAIC,IAAI,GAAG,QAAQF,QAAQ,CAACG,OAAO,CAAC,cAAc,CAAC,UAAU;MAC7DH,QAAQ,CAACI,EAAE,CAAC,MAAM,EAAEC,IAAI,IAAKH,IAAI,IAAIG,IAAK,CAAC;MAC3CL,QAAQ,CAACI,EAAE,CAAC,KAAK,EAAE,MAAMN,GAAG,CAACI,IAAI,CAAC,CAAC;IACrC,CAAC,CAAC,CACDE,EAAE,CAAC,OAAO,EAAE/B,CAAC,IAAI;MAChB0B,GAAG,CAAC,+BAA+BH,GAAG,KAAKvB,CAAC,CAACiC,OAAO,EAAE,CAAC;IACzD,CAAC,CAAC;EACN,CAAC,CAAC;AACJ,CAAC;AAED,MAAMC,mBAAmB,GAAG,MAAMC,IAAI,IAAI;EACxC,IAAIA,IAAI,CAACC,OAAO,CAACC,MAAM,KAAK,KAAK,EAAE;IACjC,MAAMC,MAAM,GAAG,MAAMhB,mBAAmB,CAACa,IAAI,CAACC,OAAO,CAACb,GAAG,CAAC;IAC1DY,IAAI,CAACI,aAAa,GAAGJ,IAAI;IACzBA,IAAI,CAACK,KAAK,GAAGF,MAAM;IACnBH,IAAI,CAACM,YAAY,GAAG,IAAI;EAC1B;EACA,OAAON,IAAI;AACb,CAAC;AAEM,MAAMO,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,CAACtC,GAAG,CAAC,yBAAyB,EAAE,IAAI,CAACyC,UAAU,CAAC;IACtDH,MAAM,CAACtC,GAAG,CAAC,kCAAkC,EAAE,IAAI,CAAC0C,eAAe,CAAC;IAEpEJ,MAAM,CAACK,IAAI,CAAC,QAAQ,EAAE,UAAUC,GAAG,EAAE1B,GAAG,EAAE2B,IAAI,EAAE;MAC9CA,IAAI,CAAC,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACC,iBAAiB,EAAE,wBAAwB,CAAC,CAAC;IAChF,CAAC,CAAC;IAEFV,MAAM,CAACK,IAAI,CACT,kBAAkB,EAClBJ,gBAAO,CAACU,GAAG,CAAC;MACVC,IAAI,EAAEA,CAAA,KAAM;QACV,OAAO,IAAI;MACb,CAAC;MACDC,KAAK,EAAEd;IACT,CAAC,CAAC;IAAE;IACJlD,WAAW,CAACiE,kBAAkB,EAC9BjE,WAAW,CAACkE,kBAAkB,EAC9B,IAAI,CAACC,aACP,CAAC;IAEDhB,MAAM,CAACiB,MAAM,CACX,kBAAkB,EAClBpE,WAAW,CAACiE,kBAAkB,EAC9BjE,WAAW,CAACkE,kBAAkB,EAC9BlE,WAAW,CAACqE,sBAAsB,EAClC,IAAI,CAACC,aACP,CAAC;IACD,OAAOnB,MAAM;EACf;EAEA,MAAMG,UAAUA,CAACG,GAAG,EAAE1B,GAAG,EAAE;IACzB,MAAMwC,MAAM,GAAGC,eAAM,CAAC3D,GAAG,CAAC4C,GAAG,CAACgB,MAAM,CAACC,KAAK,CAAC;IAC3C,IAAI,CAACH,MAAM,EAAE;MACXxC,GAAG,CAAC4C,MAAM,CAAC,GAAG,CAAC;MACf,MAAMC,GAAG,GAAG,IAAIjB,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACiB,mBAAmB,EAAE,yBAAyB,CAAC;MACvF9C,GAAG,CAAC+C,IAAI,CAAC;QAAEC,IAAI,EAAEH,GAAG,CAACG,IAAI;QAAEC,KAAK,EAAEJ,GAAG,CAACrC;MAAQ,CAAC,CAAC;MAChD;IACF;IAEA,IAAI0C,QAAQ,GAAGxB,GAAG,CAACgB,MAAM,CAACQ,QAAQ;IAClC,IAAI;MACF,MAAMC,eAAe,GAAGX,MAAM,CAACW,eAAe;MAC9C,MAAMC,IAAI,GAAG,CAAC,MAAM,MAAM,CAAC,MAAM,CAAC,EAAExE,OAAO;MAC3C,IAAIyE,WAAW,GAAGD,IAAI,CAACE,OAAO,CAACJ,QAAQ,CAAC;MACxC,IAAIxC,IAAI,GAAG,IAAIkB,aAAK,CAAC2B,IAAI,CAACL,QAAQ,EAAE;QAAErC,MAAM,EAAE;MAAG,CAAC,EAAEwC,WAAW,CAAC;MAChE,MAAMG,aAAa,GAAG,MAAM9D,QAAQ,CAAC+D,mBAAmB,CACtD/D,QAAQ,CAACgE,KAAK,CAACC,UAAU,EACzB;QAAEjD;MAAK,CAAC,EACR8B,MAAM,EACNd,GAAG,CAACkC,IACN,CAAC;MACD,IAAIJ,aAAa,EAAE9C,IAAI,EAAEmD,KAAK,EAAE;QAC9BX,QAAQ,GAAGM,aAAa,EAAE9C,IAAI,EAAEmD,KAAK;QACrCR,WAAW,GAAGD,IAAI,CAACE,OAAO,CAACJ,QAAQ,CAAC;MACtC;MAEA,IAAIY,gBAAgB,CAACpC,GAAG,EAAEyB,eAAe,CAAC,EAAE;QAC1CA,eAAe,CAACY,gBAAgB,CAACvB,MAAM,EAAEU,QAAQ,EAAExB,GAAG,EAAE1B,GAAG,EAAEqD,WAAW,CAAC,CAACW,KAAK,CAAC,MAAM;UACpFhE,GAAG,CAAC4C,MAAM,CAAC,GAAG,CAAC;UACf5C,GAAG,CAACP,GAAG,CAAC,cAAc,EAAE,YAAY,CAAC;UACrCO,GAAG,CAACiE,GAAG,CAAC,iBAAiB,CAAC;QAC5B,CAAC,CAAC;QACF;MACF;MAEA,IAAI1D,IAAI,GAAG,MAAM4C,eAAe,CAACe,WAAW,CAAC1B,MAAM,EAAEU,QAAQ,CAAC,CAACc,KAAK,CAAC,MAAM;QACzEhE,GAAG,CAAC4C,MAAM,CAAC,GAAG,CAAC;QACf5C,GAAG,CAACP,GAAG,CAAC,cAAc,EAAE,YAAY,CAAC;QACrCO,GAAG,CAACiE,GAAG,CAAC,iBAAiB,CAAC;MAC5B,CAAC,CAAC;MACF,IAAI,CAAC1D,IAAI,EAAE;QACT;MACF;MACAG,IAAI,GAAG,IAAIkB,aAAK,CAAC2B,IAAI,CAACL,QAAQ,EAAE;QAAErC,MAAM,EAAEN,IAAI,CAAC4D,QAAQ,CAAC,QAAQ;MAAE,CAAC,EAAEd,WAAW,CAAC;MACjF,MAAMe,SAAS,GAAG,MAAM1E,QAAQ,CAAC+D,mBAAmB,CAClD/D,QAAQ,CAACgE,KAAK,CAACU,SAAS,EACxB;QAAE1D,IAAI;QAAE2D,aAAa,EAAE;MAAM,CAAC,EAC9B7B,MAAM,EACNd,GAAG,CAACkC,IACN,CAAC;MAED,IAAIQ,SAAS,EAAE1D,IAAI,EAAE;QACnB2C,WAAW,GAAGD,IAAI,CAACE,OAAO,CAACc,SAAS,CAAC1D,IAAI,CAACmD,KAAK,CAAC;QAChDtD,IAAI,GAAG+D,MAAM,CAACC,IAAI,CAACH,SAAS,CAAC1D,IAAI,CAACK,KAAK,EAAE,QAAQ,CAAC;MACpD;MAEAf,GAAG,CAAC4C,MAAM,CAAC,GAAG,CAAC;MACf5C,GAAG,CAACP,GAAG,CAAC,cAAc,EAAE4D,WAAW,CAAC;MACpCrD,GAAG,CAACP,GAAG,CAAC,gBAAgB,EAAEc,IAAI,CAACiE,MAAM,CAAC;MACtC,IAAIJ,SAAS,CAACC,aAAa,EAAE;QAC3BrE,GAAG,CAACP,GAAG,CAAC,qBAAqB,EAAE,uBAAuB2E,SAAS,CAAC1D,IAAI,CAACmD,KAAK,EAAE,CAAC;MAC/E;MACA7D,GAAG,CAACiE,GAAG,CAAC1D,IAAI,CAAC;IACf,CAAC,CAAC,OAAOhC,CAAC,EAAE;MACV,MAAMsE,GAAG,GAAGnD,QAAQ,CAAC+E,YAAY,CAAClG,CAAC,EAAE;QACnCyE,IAAI,EAAEpB,aAAK,CAACC,KAAK,CAAC6C,aAAa;QAC/BlE,OAAO,EAAE,wBAAwB0C,QAAQ;MAC3C,CAAC,CAAC;MACFlD,GAAG,CAAC4C,MAAM,CAAC,GAAG,CAAC;MACf5C,GAAG,CAAC+C,IAAI,CAAC;QAAEC,IAAI,EAAEH,GAAG,CAACG,IAAI;QAAEC,KAAK,EAAEJ,GAAG,CAACrC;MAAQ,CAAC,CAAC;IAClD;EACF;EAEA,MAAM4B,aAAaA,CAACV,GAAG,EAAE1B,GAAG,EAAE2B,IAAI,EAAE;IAClC,MAAMa,MAAM,GAAGd,GAAG,CAACc,MAAM;IACzB,MAAMmC,IAAI,GAAGjD,GAAG,CAACkC,IAAI,CAACe,IAAI;IAC1B,MAAMC,QAAQ,GAAGlD,GAAG,CAACkC,IAAI,CAACgB,QAAQ;IAClC,MAAMC,QAAQ,GAAGF,IAAI,IAAI/C,aAAK,CAACkD,cAAc,CAACD,QAAQ,CAACF,IAAI,CAAC;IAC5D,IAAI,CAACC,QAAQ,IAAI,CAACpC,MAAM,CAACuC,UAAU,CAACC,sBAAsB,IAAIH,QAAQ,EAAE;MACtElD,IAAI,CACF,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACoD,eAAe,EAAE,4CAA4C,CAC3F,CAAC;MACD;IACF;IACA,IAAI,CAACL,QAAQ,IAAI,CAACpC,MAAM,CAACuC,UAAU,CAACG,0BAA0B,IAAI,CAACL,QAAQ,IAAIF,IAAI,EAAE;MACnFhD,IAAI,CACF,IAAIC,aAAK,CAACC,KAAK,CACbD,aAAK,CAACC,KAAK,CAACoD,eAAe,EAC3B,gDACF,CACF,CAAC;MACD;IACF;IACA,IAAI,CAACL,QAAQ,IAAI,CAACpC,MAAM,CAACuC,UAAU,CAACI,eAAe,IAAI,CAACR,IAAI,EAAE;MAC5DhD,IAAI,CAAC,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACoD,eAAe,EAAE,oCAAoC,CAAC,CAAC;MACxF;IACF;IACA,MAAM9B,eAAe,GAAGX,MAAM,CAACW,eAAe;IAC9C,MAAM;MAAED;IAAS,CAAC,GAAGxB,GAAG,CAACgB,MAAM;IAC/B,MAAMW,WAAW,GAAG3B,GAAG,CAAC5C,GAAG,CAAC,cAAc,CAAC;IAE3C,IAAI,CAAC4C,GAAG,CAACtB,IAAI,IAAI,CAACsB,GAAG,CAACtB,IAAI,CAACoE,MAAM,EAAE;MACjC7C,IAAI,CAAC,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACoD,eAAe,EAAE,sBAAsB,CAAC,CAAC;MAC1E;IACF;IAEA,MAAMhC,KAAK,GAAGE,eAAe,CAACiC,gBAAgB,CAAClC,QAAQ,CAAC;IACxD,IAAID,KAAK,EAAE;MACTtB,IAAI,CAACsB,KAAK,CAAC;MACX;IACF;IAEA,MAAMoC,cAAc,GAAG7C,MAAM,CAACuC,UAAU,EAAEM,cAAc;IACxD,IAAI,CAACT,QAAQ,IAAIS,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,GAAGlC,WAAW;MAC3B,IAAIH,QAAQ,IAAIA,QAAQ,CAAC2C,QAAQ,CAAC,GAAG,CAAC,EAAE;QACtCN,SAAS,GAAGrC,QAAQ,CAAC4C,SAAS,CAAC5C,QAAQ,CAAC6C,WAAW,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;MAC/D,CAAC,MAAM,IAAI1C,WAAW,IAAIA,WAAW,CAACwC,QAAQ,CAAC,GAAG,CAAC,EAAE;QACnDN,SAAS,GAAGlC,WAAW,CAAC2C,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;QAC7C5D,IAAI,CACF,IAAIC,aAAK,CAACC,KAAK,CACbD,aAAK,CAACC,KAAK,CAACoD,eAAe,EAC3B,4BAA4BM,SAAS,eACvC,CACF,CAAC;QACD;MACF;IACF;IAEA,MAAM1E,MAAM,GAAGa,GAAG,CAACtB,IAAI,CAAC+D,QAAQ,CAAC,QAAQ,CAAC;IAC1C,MAAMzD,IAAI,GAAG,IAAIkB,aAAK,CAAC2B,IAAI,CAACL,QAAQ,EAAE;MAAErC;IAAO,CAAC,EAAEwC,WAAW,CAAC;IAC9D,MAAM;MAAE6C,QAAQ,GAAG,CAAC,CAAC;MAAEC,IAAI,GAAG,CAAC;IAAE,CAAC,GAAGzE,GAAG,CAAC0E,QAAQ,IAAI,CAAC,CAAC;IACvD,IAAI;MACF;MACAxG,KAAK,CAACyG,uBAAuB,CAAC7D,MAAM,EAAE0D,QAAQ,CAAC;MAC/CtG,KAAK,CAACyG,uBAAuB,CAAC7D,MAAM,EAAE2D,IAAI,CAAC;IAC7C,CAAC,CAAC,OAAOlD,KAAK,EAAE;MACdtB,IAAI,CAAC,IAAIC,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACyE,gBAAgB,EAAErD,KAAK,CAAC,CAAC;MAC1D;IACF;IACAvC,IAAI,CAAC6F,OAAO,CAACJ,IAAI,CAAC;IAClBzF,IAAI,CAAC8F,WAAW,CAACN,QAAQ,CAAC;IAC1B,MAAMO,QAAQ,GAAGnC,MAAM,CAACoC,UAAU,CAAChF,GAAG,CAACtB,IAAI,CAAC;IAC5C,MAAMuG,UAAU,GAAG;MAAEjG,IAAI;MAAE+F;IAAS,CAAC;IACrC,IAAI;MACF;MACA,MAAMjD,aAAa,GAAG,MAAM9D,QAAQ,CAAC+D,mBAAmB,CACtD/D,QAAQ,CAACgE,KAAK,CAACkD,UAAU,EACzBD,UAAU,EACVnE,MAAM,EACNd,GAAG,CAACkC,IACN,CAAC;MACD,IAAIiD,UAAU;MACd;MACA,IAAIrD,aAAa,YAAY5B,aAAK,CAAC2B,IAAI,EAAE;QACvCoD,UAAU,CAACjG,IAAI,GAAG8C,aAAa;QAC/B,IAAIA,aAAa,CAACsD,GAAG,CAAC,CAAC,EAAE;UACvB;UACAH,UAAU,CAACF,QAAQ,GAAG,IAAI;UAC1BI,UAAU,GAAG;YACXC,GAAG,EAAEtD,aAAa,CAACsD,GAAG,CAAC,CAAC;YACxBC,IAAI,EAAEvD,aAAa,CAACK;UACtB,CAAC;QACH;MACF;MACA;MACA,IAAI,CAACgD,UAAU,EAAE;QACf;QACA,MAAMpG,mBAAmB,CAACkG,UAAU,CAACjG,IAAI,CAAC;QAC1C;QACA,MAAMsG,UAAU,GAAG1C,MAAM,CAACC,IAAI,CAACoC,UAAU,CAACjG,IAAI,CAACK,KAAK,EAAE,QAAQ,CAAC;QAC/D4F,UAAU,CAACF,QAAQ,GAAGnC,MAAM,CAACoC,UAAU,CAACM,UAAU,CAAC;QACnD;QACA,MAAMC,WAAW,GAAG;UAClBf,QAAQ,EAAES,UAAU,CAACjG,IAAI,CAACwG;QAC5B,CAAC;QACD;QACA;QACA,MAAMC,QAAQ,GACZjI,MAAM,CAACkI,IAAI,CAACT,UAAU,CAACjG,IAAI,CAAC2G,KAAK,CAAC,CAAC7C,MAAM,GAAG,CAAC,GAAG;UAAE2B,IAAI,EAAEQ,UAAU,CAACjG,IAAI,CAAC2G;QAAM,CAAC,GAAG,CAAC,CAAC;QACtFnI,MAAM,CAACoI,MAAM,CAACL,WAAW,EAAEE,QAAQ,CAAC;QACpC;QACA,MAAMI,gBAAgB,GAAG,MAAMpE,eAAe,CAACqE,UAAU,CACvDhF,MAAM,EACNmE,UAAU,CAACjG,IAAI,CAACmD,KAAK,EACrBmD,UAAU,EACVL,UAAU,CAACjG,IAAI,CAACC,OAAO,CAACqB,IAAI,EAC5BiF,WACF,CAAC;QACD;QACAN,UAAU,CAACjG,IAAI,CAACmD,KAAK,GAAG0D,gBAAgB,CAACR,IAAI;QAC7CJ,UAAU,CAACjG,IAAI,CAAC+G,IAAI,GAAGF,gBAAgB,CAACT,GAAG;QAC3CH,UAAU,CAACjG,IAAI,CAACM,YAAY,GAAG,IAAI;QACnC2F,UAAU,CAACjG,IAAI,CAACI,aAAa,GAAGf,OAAO,CAAC2H,OAAO,CAACf,UAAU,CAACjG,IAAI,CAAC;QAChEmG,UAAU,GAAG;UACXC,GAAG,EAAES,gBAAgB,CAACT,GAAG;UACzBC,IAAI,EAAEQ,gBAAgB,CAACR;QACzB,CAAC;MACH;MACA;MACA,MAAMrH,QAAQ,CAAC+D,mBAAmB,CAAC/D,QAAQ,CAACgE,KAAK,CAACiE,SAAS,EAAEhB,UAAU,EAAEnE,MAAM,EAAEd,GAAG,CAACkC,IAAI,CAAC;MAC1F5D,GAAG,CAAC4C,MAAM,CAAC,GAAG,CAAC;MACf5C,GAAG,CAACP,GAAG,CAAC,UAAU,EAAEoH,UAAU,CAACC,GAAG,CAAC;MACnC9G,GAAG,CAAC+C,IAAI,CAAC8D,UAAU,CAAC;IACtB,CAAC,CAAC,OAAOtI,CAAC,EAAE;MACVqJ,eAAM,CAAC3E,KAAK,CAAC,yBAAyB,EAAE1E,CAAC,CAAC;MAC1C,MAAM0E,KAAK,GAAGvD,QAAQ,CAAC+E,YAAY,CAAClG,CAAC,EAAE;QACrCyE,IAAI,EAAEpB,aAAK,CAACC,KAAK,CAACoD,eAAe;QACjCzE,OAAO,EAAE,yBAAyBmG,UAAU,CAACjG,IAAI,CAACmD,KAAK;MACzD,CAAC,CAAC;MACFlC,IAAI,CAACsB,KAAK,CAAC;IACb;EACF;EAEA,MAAMV,aAAaA,CAACb,GAAG,EAAE1B,GAAG,EAAE2B,IAAI,EAAE;IAClC,IAAI;MACF,MAAM;QAAEwB;MAAgB,CAAC,GAAGzB,GAAG,CAACc,MAAM;MACtC,MAAM;QAAEU;MAAS,CAAC,GAAGxB,GAAG,CAACgB,MAAM;MAC/B;MACA,MAAMhC,IAAI,GAAG,IAAIkB,aAAK,CAAC2B,IAAI,CAACL,QAAQ,CAAC;MACrCxC,IAAI,CAAC+G,IAAI,GAAG,MAAMtE,eAAe,CAAC0E,OAAO,CAACC,eAAe,CAACpG,GAAG,CAACc,MAAM,EAAEU,QAAQ,CAAC;MAC/E,MAAMyD,UAAU,GAAG;QAAEjG,IAAI;QAAE+F,QAAQ,EAAE;MAAK,CAAC;MAC3C,MAAM/G,QAAQ,CAAC+D,mBAAmB,CAChC/D,QAAQ,CAACgE,KAAK,CAACqE,YAAY,EAC3BpB,UAAU,EACVjF,GAAG,CAACc,MAAM,EACVd,GAAG,CAACkC,IACN,CAAC;MACD;MACA,MAAMT,eAAe,CAAC6E,UAAU,CAACtG,GAAG,CAACc,MAAM,EAAEU,QAAQ,CAAC;MACtD;MACA,MAAMxD,QAAQ,CAAC+D,mBAAmB,CAChC/D,QAAQ,CAACgE,KAAK,CAACuE,WAAW,EAC1BtB,UAAU,EACVjF,GAAG,CAACc,MAAM,EACVd,GAAG,CAACkC,IACN,CAAC;MACD5D,GAAG,CAAC4C,MAAM,CAAC,GAAG,CAAC;MACf;MACA5C,GAAG,CAACiE,GAAG,CAAC,CAAC;IACX,CAAC,CAAC,OAAO1F,CAAC,EAAE;MACVqJ,eAAM,CAAC3E,KAAK,CAAC,yBAAyB,EAAE1E,CAAC,CAAC;MAC1C,MAAM0E,KAAK,GAAGvD,QAAQ,CAAC+E,YAAY,CAAClG,CAAC,EAAE;QACrCyE,IAAI,EAAEpB,aAAK,CAACC,KAAK,CAACqG,iBAAiB;QACnC1H,OAAO,EAAE;MACX,CAAC,CAAC;MACFmB,IAAI,CAACsB,KAAK,CAAC;IACb;EACF;EAEA,MAAMzB,eAAeA,CAACE,GAAG,EAAE1B,GAAG,EAAE;IAC9B,IAAI;MACF,MAAMwC,MAAM,GAAGC,eAAM,CAAC3D,GAAG,CAAC4C,GAAG,CAACgB,MAAM,CAACC,KAAK,CAAC;MAC3C,MAAM;QAAEQ;MAAgB,CAAC,GAAGX,MAAM;MAClC,MAAM;QAAEU;MAAS,CAAC,GAAGxB,GAAG,CAACgB,MAAM;MAC/B,MAAMnC,IAAI,GAAG,MAAM4C,eAAe,CAACgF,WAAW,CAACjF,QAAQ,CAAC;MACxDlD,GAAG,CAAC4C,MAAM,CAAC,GAAG,CAAC;MACf5C,GAAG,CAAC+C,IAAI,CAACxC,IAAI,CAAC;IAChB,CAAC,CAAC,OAAOhC,CAAC,EAAE;MACVyB,GAAG,CAAC4C,MAAM,CAAC,GAAG,CAAC;MACf5C,GAAG,CAAC+C,IAAI,CAAC,CAAC,CAAC,CAAC;IACd;EACF;AACF;AAACqF,OAAA,CAAAnH,WAAA,GAAAA,WAAA;AAED,SAAS6C,gBAAgBA,CAACpC,GAAG,EAAEyB,eAAe,EAAE;EAC9C,MAAMkF,KAAK,GAAG,CAAC3G,GAAG,CAAC5C,GAAG,CAAC,OAAO,CAAC,IAAI,KAAK,EAAEkH,KAAK,CAAC,GAAG,CAAC;EACpD,MAAMsC,KAAK,GAAGC,MAAM,CAACF,KAAK,CAAC,CAAC,CAAC,CAAC;EAC9B,MAAMpE,GAAG,GAAGsE,MAAM,CAACF,KAAK,CAAC,CAAC,CAAC,CAAC;EAC5B,OACE,CAAC,CAACG,KAAK,CAACF,KAAK,CAAC,IAAI,CAACE,KAAK,CAACvE,GAAG,CAAC,KAAK,OAAOd,eAAe,CAAC0E,OAAO,CAAC9D,gBAAgB,KAAK,UAAU;AAEpG","ignoreList":[]}
|
package/lib/TestUtils.js
CHANGED
|
@@ -3,7 +3,9 @@
|
|
|
3
3
|
Object.defineProperty(exports, "__esModule", {
|
|
4
4
|
value: true
|
|
5
5
|
});
|
|
6
|
+
exports.Connections = void 0;
|
|
6
7
|
exports.destroyAllDataPermanently = destroyAllDataPermanently;
|
|
8
|
+
exports.getConnectionsCount = getConnectionsCount;
|
|
7
9
|
exports.resolvingPromise = resolvingPromise;
|
|
8
10
|
exports.sleep = sleep;
|
|
9
11
|
var _cache = _interopRequireDefault(require("./cache"));
|
|
@@ -46,4 +48,40 @@ function resolvingPromise() {
|
|
|
46
48
|
function sleep(ms) {
|
|
47
49
|
return new Promise(resolve => setTimeout(resolve, ms));
|
|
48
50
|
}
|
|
49
|
-
|
|
51
|
+
function getConnectionsCount(server) {
|
|
52
|
+
return new Promise((resolve, reject) => {
|
|
53
|
+
server.getConnections((err, count) => {
|
|
54
|
+
/* istanbul ignore next */
|
|
55
|
+
if (err) {
|
|
56
|
+
reject(err);
|
|
57
|
+
} else {
|
|
58
|
+
resolve(count);
|
|
59
|
+
}
|
|
60
|
+
});
|
|
61
|
+
});
|
|
62
|
+
}
|
|
63
|
+
;
|
|
64
|
+
class Connections {
|
|
65
|
+
constructor() {
|
|
66
|
+
this.sockets = new Set();
|
|
67
|
+
}
|
|
68
|
+
track(server) {
|
|
69
|
+
server.on('connection', socket => {
|
|
70
|
+
this.sockets.add(socket);
|
|
71
|
+
socket.on('close', () => {
|
|
72
|
+
this.sockets.delete(socket);
|
|
73
|
+
});
|
|
74
|
+
});
|
|
75
|
+
}
|
|
76
|
+
destroyAll() {
|
|
77
|
+
for (const socket of this.sockets.values()) {
|
|
78
|
+
socket.destroy();
|
|
79
|
+
}
|
|
80
|
+
this.sockets.clear();
|
|
81
|
+
}
|
|
82
|
+
count() {
|
|
83
|
+
return this.sockets.size;
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
exports.Connections = Connections;
|
|
87
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJfY2FjaGUiLCJfaW50ZXJvcFJlcXVpcmVEZWZhdWx0IiwicmVxdWlyZSIsIl9TY2hlbWFDYWNoZSIsImUiLCJfX2VzTW9kdWxlIiwiZGVmYXVsdCIsImRlc3Ryb3lBbGxEYXRhUGVybWFuZW50bHkiLCJmYXN0IiwicHJvY2VzcyIsImVudiIsIlRFU1RJTkciLCJQcm9taXNlIiwiYWxsIiwiT2JqZWN0Iiwia2V5cyIsIkFwcENhY2hlIiwiY2FjaGUiLCJtYXAiLCJhcHBJZCIsImFwcCIsImdldCIsImRlbGV0ZVByb21pc2VzIiwiY2FjaGVBZGFwdGVyIiwiY2xlYXIiLCJwdXNoIiwiZGF0YWJhc2VDb250cm9sbGVyIiwiZGVsZXRlRXZlcnl0aGluZyIsImRhdGFiYXNlQWRhcHRlciIsIlNjaGVtYUNhY2hlIiwiZGVsZXRlQWxsQ2xhc3NlcyIsInJlc29sdmluZ1Byb21pc2UiLCJyZXMiLCJyZWoiLCJwcm9taXNlIiwicmVzb2x2ZSIsInJlamVjdCIsInNsZWVwIiwibXMiLCJzZXRUaW1lb3V0IiwiZ2V0Q29ubmVjdGlvbnNDb3VudCIsInNlcnZlciIsImdldENvbm5lY3Rpb25zIiwiZXJyIiwiY291bnQiLCJDb25uZWN0aW9ucyIsImNvbnN0cnVjdG9yIiwic29ja2V0cyIsIlNldCIsInRyYWNrIiwib24iLCJzb2NrZXQiLCJhZGQiLCJkZWxldGUiLCJkZXN0cm95QWxsIiwidmFsdWVzIiwiZGVzdHJveSIsInNpemUiLCJleHBvcnRzIl0sInNvdXJjZXMiOlsiLi4vc3JjL1Rlc3RVdGlscy5qcyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgQXBwQ2FjaGUgZnJvbSAnLi9jYWNoZSc7XG5pbXBvcnQgU2NoZW1hQ2FjaGUgZnJvbSAnLi9BZGFwdGVycy9DYWNoZS9TY2hlbWFDYWNoZSc7XG5cbi8qKlxuICogRGVzdHJveXMgYWxsIGRhdGEgaW4gdGhlIGRhdGFiYXNlXG4gKiBAcGFyYW0ge2Jvb2xlYW59IGZhc3Qgc2V0IHRvIHRydWUgaWYgaXQncyBvayB0byBqdXN0IGRyb3Agb2JqZWN0cyBhbmQgbm90IGluZGV4ZXMuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBkZXN0cm95QWxsRGF0YVBlcm1hbmVudGx5KGZhc3QpIHtcbiAgaWYgKCFwcm9jZXNzLmVudi5URVNUSU5HKSB7XG4gICAgdGhyb3cgJ09ubHkgc3VwcG9ydGVkIGluIHRlc3QgZW52aXJvbm1lbnQnO1xuICB9XG4gIHJldHVybiBQcm9taXNlLmFsbChcbiAgICBPYmplY3Qua2V5cyhBcHBDYWNoZS5jYWNoZSkubWFwKGFwcElkID0+IHtcbiAgICAgIGNvbnN0IGFwcCA9IEFwcENhY2hlLmdldChhcHBJZCk7XG4gICAgICBjb25zdCBkZWxldGVQcm9taXNlcyA9IFtdO1xuICAgICAgaWYgKGFwcC5jYWNoZUFkYXB0ZXIgJiYgYXBwLmNhY2hlQWRhcHRlci5jbGVhcikge1xuICAgICAgICBkZWxldGVQcm9taXNlcy5wdXNoKGFwcC5jYWNoZUFkYXB0ZXIuY2xlYXIoKSk7XG4gICAgICB9XG4gICAgICBpZiAoYXBwLmRhdGFiYXNlQ29udHJvbGxlcikge1xuICAgICAgICBkZWxldGVQcm9taXNlcy5wdXNoKGFwcC5kYXRhYmFzZUNvbnRyb2xsZXIuZGVsZXRlRXZlcnl0aGluZyhmYXN0KSk7XG4gICAgICB9IGVsc2UgaWYgKGFwcC5kYXRhYmFzZUFkYXB0ZXIpIHtcbiAgICAgICAgU2NoZW1hQ2FjaGUuY2xlYXIoKTtcbiAgICAgICAgZGVsZXRlUHJvbWlzZXMucHVzaChhcHAuZGF0YWJhc2VBZGFwdGVyLmRlbGV0ZUFsbENsYXNzZXMoZmFzdCkpO1xuICAgICAgfVxuICAgICAgcmV0dXJuIFByb21pc2UuYWxsKGRlbGV0ZVByb21pc2VzKTtcbiAgICB9KVxuICApO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcmVzb2x2aW5nUHJvbWlzZSgpIHtcbiAgbGV0IHJlcztcbiAgbGV0IHJlajtcbiAgY29uc3QgcHJvbWlzZSA9IG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICByZXMgPSByZXNvbHZlO1xuICAgIHJlaiA9IHJlamVjdDtcbiAgfSk7XG4gIHByb21pc2UucmVzb2x2ZSA9IHJlcztcbiAgcHJvbWlzZS5yZWplY3QgPSByZWo7XG4gIHJldHVybiBwcm9taXNlO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gc2xlZXAobXMpIHtcbiAgcmV0dXJuIG5ldyBQcm9taXNlKChyZXNvbHZlKSA9PiBzZXRUaW1lb3V0KHJlc29sdmUsIG1zKSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRDb25uZWN0aW9uc0NvdW50KHNlcnZlcikge1xuICByZXR1cm4gbmV3IFByb21pc2UoKHJlc29sdmUsIHJlamVjdCkgPT4ge1xuICAgIHNlcnZlci5nZXRDb25uZWN0aW9ucygoZXJyLCBjb3VudCkgPT4ge1xuICAgICAgLyogaXN0YW5idWwgaWdub3JlIG5leHQgKi9cbiAgICAgIGlmIChlcnIpIHtcbiAgICAgICAgcmVqZWN0KGVycik7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICByZXNvbHZlKGNvdW50KTtcbiAgICAgIH1cbiAgICB9KTtcbiAgfSk7XG59O1xuXG5leHBvcnQgY2xhc3MgQ29ubmVjdGlvbnMge1xuICBjb25zdHJ1Y3RvcigpIHtcbiAgICB0aGlzLnNvY2tldHMgPSBuZXcgU2V0KCk7XG4gIH1cblxuICB0cmFjayhzZXJ2ZXIpIHtcbiAgICBzZXJ2ZXIub24oJ2Nvbm5lY3Rpb24nLCBzb2NrZXQgPT4ge1xuICAgICAgdGhpcy5zb2NrZXRzLmFkZChzb2NrZXQpO1xuICAgICAgc29ja2V0Lm9uKCdjbG9zZScsICgpID0+IHtcbiAgICAgICAgdGhpcy5zb2NrZXRzLmRlbGV0ZShzb2NrZXQpO1xuICAgICAgfSk7XG4gICAgfSk7XG4gIH1cblxuICBkZXN0cm95QWxsKCkge1xuICAgIGZvciAoY29uc3Qgc29ja2V0IG9mIHRoaXMuc29ja2V0cy52YWx1ZXMoKSkge1xuICAgICAgc29ja2V0LmRlc3Ryb3koKTtcbiAgICB9XG4gICAgdGhpcy5zb2NrZXRzLmNsZWFyKCk7XG4gIH1c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
|