parse-server 2.8.4 → 8.6.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (240) hide show
  1. package/LICENSE +167 -25
  2. package/NOTICE +10 -0
  3. package/README.md +929 -278
  4. package/lib/AccountLockout.js +47 -30
  5. package/lib/Adapters/AdapterLoader.js +21 -6
  6. package/lib/Adapters/Analytics/AnalyticsAdapter.js +15 -12
  7. package/lib/Adapters/Auth/AuthAdapter.js +116 -13
  8. package/lib/Adapters/Auth/BaseCodeAuthAdapter.js +99 -0
  9. package/lib/Adapters/Auth/OAuth1Client.js +27 -46
  10. package/lib/Adapters/Auth/apple.js +123 -0
  11. package/lib/Adapters/Auth/facebook.js +162 -35
  12. package/lib/Adapters/Auth/gcenter.js +217 -0
  13. package/lib/Adapters/Auth/github.js +118 -48
  14. package/lib/Adapters/Auth/google.js +160 -51
  15. package/lib/Adapters/Auth/gpgames.js +125 -0
  16. package/lib/Adapters/Auth/httpsRequest.js +6 -7
  17. package/lib/Adapters/Auth/index.js +170 -62
  18. package/lib/Adapters/Auth/instagram.js +114 -40
  19. package/lib/Adapters/Auth/janraincapture.js +52 -23
  20. package/lib/Adapters/Auth/janrainengage.js +19 -36
  21. package/lib/Adapters/Auth/keycloak.js +148 -0
  22. package/lib/Adapters/Auth/ldap.js +167 -0
  23. package/lib/Adapters/Auth/line.js +125 -0
  24. package/lib/Adapters/Auth/linkedin.js +111 -55
  25. package/lib/Adapters/Auth/meetup.js +24 -34
  26. package/lib/Adapters/Auth/mfa.js +324 -0
  27. package/lib/Adapters/Auth/microsoft.js +111 -0
  28. package/lib/Adapters/Auth/oauth2.js +97 -162
  29. package/lib/Adapters/Auth/phantauth.js +53 -0
  30. package/lib/Adapters/Auth/qq.js +108 -49
  31. package/lib/Adapters/Auth/spotify.js +107 -55
  32. package/lib/Adapters/Auth/twitter.js +188 -48
  33. package/lib/Adapters/Auth/utils.js +28 -0
  34. package/lib/Adapters/Auth/vkontakte.js +26 -39
  35. package/lib/Adapters/Auth/wechat.js +106 -44
  36. package/lib/Adapters/Auth/weibo.js +132 -58
  37. package/lib/Adapters/Cache/CacheAdapter.js +13 -8
  38. package/lib/Adapters/Cache/InMemoryCache.js +3 -13
  39. package/lib/Adapters/Cache/InMemoryCacheAdapter.js +5 -13
  40. package/lib/Adapters/Cache/LRUCache.js +13 -27
  41. package/lib/Adapters/Cache/NullCacheAdapter.js +3 -8
  42. package/lib/Adapters/Cache/RedisCacheAdapter.js +85 -76
  43. package/lib/Adapters/Cache/SchemaCache.js +25 -0
  44. package/lib/Adapters/Email/MailAdapter.js +10 -8
  45. package/lib/Adapters/Files/FilesAdapter.js +83 -25
  46. package/lib/Adapters/Files/GridFSBucketAdapter.js +231 -0
  47. package/lib/Adapters/Files/GridStoreAdapter.js +4 -91
  48. package/lib/Adapters/Logger/LoggerAdapter.js +18 -14
  49. package/lib/Adapters/Logger/WinstonLogger.js +69 -88
  50. package/lib/Adapters/Logger/WinstonLoggerAdapter.js +7 -16
  51. package/lib/Adapters/MessageQueue/EventEmitterMQ.js +8 -26
  52. package/lib/Adapters/PubSub/EventEmitterPubSub.js +12 -25
  53. package/lib/Adapters/PubSub/PubSubAdapter.js +34 -0
  54. package/lib/Adapters/PubSub/RedisPubSub.js +42 -19
  55. package/lib/Adapters/Push/PushAdapter.js +14 -7
  56. package/lib/Adapters/Storage/Mongo/MongoCollection.js +137 -45
  57. package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +158 -63
  58. package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +320 -168
  59. package/lib/Adapters/Storage/Mongo/MongoTransform.js +279 -306
  60. package/lib/Adapters/Storage/Postgres/PostgresClient.js +14 -10
  61. package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +47 -21
  62. package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +854 -468
  63. package/lib/Adapters/Storage/Postgres/sql/index.js +4 -6
  64. package/lib/Adapters/Storage/StorageAdapter.js +1 -1
  65. package/lib/Adapters/WebSocketServer/WSAdapter.js +35 -0
  66. package/lib/Adapters/WebSocketServer/WSSAdapter.js +66 -0
  67. package/lib/Auth.js +488 -125
  68. package/lib/ClientSDK.js +2 -6
  69. package/lib/Config.js +525 -94
  70. package/lib/Controllers/AdaptableController.js +5 -25
  71. package/lib/Controllers/AnalyticsController.js +22 -23
  72. package/lib/Controllers/CacheController.js +10 -31
  73. package/lib/Controllers/DatabaseController.js +767 -313
  74. package/lib/Controllers/FilesController.js +49 -54
  75. package/lib/Controllers/HooksController.js +80 -84
  76. package/lib/Controllers/LiveQueryController.js +35 -22
  77. package/lib/Controllers/LoggerController.js +22 -58
  78. package/lib/Controllers/ParseGraphQLController.js +293 -0
  79. package/lib/Controllers/PushController.js +58 -49
  80. package/lib/Controllers/SchemaController.js +916 -422
  81. package/lib/Controllers/UserController.js +265 -180
  82. package/lib/Controllers/index.js +90 -125
  83. package/lib/Controllers/types.js +1 -1
  84. package/lib/Deprecator/Deprecations.js +30 -0
  85. package/lib/Deprecator/Deprecator.js +127 -0
  86. package/lib/Error.js +48 -0
  87. package/lib/GraphQL/ParseGraphQLSchema.js +375 -0
  88. package/lib/GraphQL/ParseGraphQLServer.js +214 -0
  89. package/lib/GraphQL/helpers/objectsMutations.js +30 -0
  90. package/lib/GraphQL/helpers/objectsQueries.js +246 -0
  91. package/lib/GraphQL/loaders/configMutations.js +87 -0
  92. package/lib/GraphQL/loaders/configQueries.js +79 -0
  93. package/lib/GraphQL/loaders/defaultGraphQLMutations.js +21 -0
  94. package/lib/GraphQL/loaders/defaultGraphQLQueries.js +23 -0
  95. package/lib/GraphQL/loaders/defaultGraphQLTypes.js +1098 -0
  96. package/lib/GraphQL/loaders/defaultRelaySchema.js +53 -0
  97. package/lib/GraphQL/loaders/filesMutations.js +107 -0
  98. package/lib/GraphQL/loaders/functionsMutations.js +78 -0
  99. package/lib/GraphQL/loaders/parseClassMutations.js +268 -0
  100. package/lib/GraphQL/loaders/parseClassQueries.js +127 -0
  101. package/lib/GraphQL/loaders/parseClassTypes.js +493 -0
  102. package/lib/GraphQL/loaders/schemaDirectives.js +62 -0
  103. package/lib/GraphQL/loaders/schemaMutations.js +162 -0
  104. package/lib/GraphQL/loaders/schemaQueries.js +81 -0
  105. package/lib/GraphQL/loaders/schemaTypes.js +341 -0
  106. package/lib/GraphQL/loaders/usersMutations.js +433 -0
  107. package/lib/GraphQL/loaders/usersQueries.js +90 -0
  108. package/lib/GraphQL/parseGraphQLUtils.js +63 -0
  109. package/lib/GraphQL/transformers/className.js +14 -0
  110. package/lib/GraphQL/transformers/constraintType.js +53 -0
  111. package/lib/GraphQL/transformers/inputType.js +51 -0
  112. package/lib/GraphQL/transformers/mutation.js +274 -0
  113. package/lib/GraphQL/transformers/outputType.js +51 -0
  114. package/lib/GraphQL/transformers/query.js +237 -0
  115. package/lib/GraphQL/transformers/schemaFields.js +99 -0
  116. package/lib/KeyPromiseQueue.js +48 -0
  117. package/lib/LiveQuery/Client.js +25 -33
  118. package/lib/LiveQuery/Id.js +2 -5
  119. package/lib/LiveQuery/ParseCloudCodePublisher.js +26 -23
  120. package/lib/LiveQuery/ParseLiveQueryServer.js +560 -285
  121. package/lib/LiveQuery/ParsePubSub.js +7 -16
  122. package/lib/LiveQuery/ParseWebSocketServer.js +42 -39
  123. package/lib/LiveQuery/QueryTools.js +76 -15
  124. package/lib/LiveQuery/RequestSchema.js +111 -97
  125. package/lib/LiveQuery/SessionTokenCache.js +23 -36
  126. package/lib/LiveQuery/Subscription.js +8 -17
  127. package/lib/LiveQuery/equalObjects.js +2 -3
  128. package/lib/Options/Definitions.js +1355 -382
  129. package/lib/Options/docs.js +301 -62
  130. package/lib/Options/index.js +11 -1
  131. package/lib/Options/parsers.js +14 -10
  132. package/lib/Page.js +44 -0
  133. package/lib/ParseMessageQueue.js +6 -13
  134. package/lib/ParseServer.js +474 -235
  135. package/lib/ParseServerRESTController.js +102 -40
  136. package/lib/PromiseRouter.js +39 -50
  137. package/lib/Push/PushQueue.js +24 -30
  138. package/lib/Push/PushWorker.js +32 -56
  139. package/lib/Push/utils.js +22 -35
  140. package/lib/RestQuery.js +361 -139
  141. package/lib/RestWrite.js +713 -344
  142. package/lib/Routers/AggregateRouter.js +97 -71
  143. package/lib/Routers/AnalyticsRouter.js +8 -14
  144. package/lib/Routers/AudiencesRouter.js +16 -35
  145. package/lib/Routers/ClassesRouter.js +86 -72
  146. package/lib/Routers/CloudCodeRouter.js +28 -37
  147. package/lib/Routers/FeaturesRouter.js +22 -25
  148. package/lib/Routers/FilesRouter.js +266 -171
  149. package/lib/Routers/FunctionsRouter.js +87 -103
  150. package/lib/Routers/GlobalConfigRouter.js +94 -33
  151. package/lib/Routers/GraphQLRouter.js +41 -0
  152. package/lib/Routers/HooksRouter.js +43 -47
  153. package/lib/Routers/IAPValidationRouter.js +57 -70
  154. package/lib/Routers/InstallationsRouter.js +17 -25
  155. package/lib/Routers/LogsRouter.js +10 -25
  156. package/lib/Routers/PagesRouter.js +647 -0
  157. package/lib/Routers/PublicAPIRouter.js +104 -112
  158. package/lib/Routers/PurgeRouter.js +19 -29
  159. package/lib/Routers/PushRouter.js +14 -28
  160. package/lib/Routers/RolesRouter.js +7 -14
  161. package/lib/Routers/SchemasRouter.js +63 -42
  162. package/lib/Routers/SecurityRouter.js +34 -0
  163. package/lib/Routers/SessionsRouter.js +25 -38
  164. package/lib/Routers/UsersRouter.js +463 -190
  165. package/lib/SchemaMigrations/DefinedSchemas.js +379 -0
  166. package/lib/SchemaMigrations/Migrations.js +30 -0
  167. package/lib/Security/Check.js +109 -0
  168. package/lib/Security/CheckGroup.js +44 -0
  169. package/lib/Security/CheckGroups/CheckGroupDatabase.js +44 -0
  170. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +96 -0
  171. package/lib/Security/CheckGroups/CheckGroups.js +21 -0
  172. package/lib/Security/CheckRunner.js +213 -0
  173. package/lib/SharedRest.js +29 -0
  174. package/lib/StatusHandler.js +96 -93
  175. package/lib/TestUtils.js +70 -14
  176. package/lib/Utils.js +468 -0
  177. package/lib/batch.js +74 -40
  178. package/lib/cache.js +8 -8
  179. package/lib/cli/definitions/parse-live-query-server.js +4 -3
  180. package/lib/cli/definitions/parse-server.js +4 -3
  181. package/lib/cli/parse-live-query-server.js +9 -17
  182. package/lib/cli/parse-server.js +49 -47
  183. package/lib/cli/utils/commander.js +20 -29
  184. package/lib/cli/utils/runner.js +31 -32
  185. package/lib/cloud-code/Parse.Cloud.js +711 -36
  186. package/lib/cloud-code/Parse.Server.js +21 -0
  187. package/lib/cryptoUtils.js +6 -11
  188. package/lib/defaults.js +21 -15
  189. package/lib/deprecated.js +1 -1
  190. package/lib/index.js +78 -67
  191. package/lib/logger.js +12 -20
  192. package/lib/middlewares.js +484 -160
  193. package/lib/password.js +10 -6
  194. package/lib/request.js +175 -0
  195. package/lib/requiredParameter.js +4 -3
  196. package/lib/rest.js +157 -82
  197. package/lib/triggers.js +627 -185
  198. package/lib/vendor/README.md +3 -3
  199. package/lib/vendor/mongodbUrl.js +224 -137
  200. package/package.json +135 -57
  201. package/postinstall.js +38 -50
  202. package/public_html/invalid_verification_link.html +3 -3
  203. package/types/@types/@parse/fs-files-adapter/index.d.ts +5 -0
  204. package/types/@types/deepcopy/index.d.ts +5 -0
  205. package/types/LiveQuery/ParseLiveQueryServer.d.ts +40 -0
  206. package/types/Options/index.d.ts +301 -0
  207. package/types/ParseServer.d.ts +65 -0
  208. package/types/eslint.config.mjs +30 -0
  209. package/types/index.d.ts +21 -0
  210. package/types/logger.d.ts +2 -0
  211. package/types/tests.ts +44 -0
  212. package/types/tsconfig.json +24 -0
  213. package/CHANGELOG.md +0 -1246
  214. package/PATENTS +0 -37
  215. package/bin/dev +0 -37
  216. package/lib/.DS_Store +0 -0
  217. package/lib/Adapters/Auth/common.js +0 -2
  218. package/lib/Adapters/Auth/facebookaccountkit.js +0 -69
  219. package/lib/Controllers/SchemaCache.js +0 -97
  220. package/lib/LiveQuery/.DS_Store +0 -0
  221. package/lib/cli/utils/parsers.js +0 -77
  222. package/lib/cloud-code/.DS_Store +0 -0
  223. package/lib/cloud-code/HTTPResponse.js +0 -57
  224. package/lib/cloud-code/Untitled-1 +0 -123
  225. package/lib/cloud-code/httpRequest.js +0 -102
  226. package/lib/cloud-code/team.html +0 -123
  227. package/lib/graphql/ParseClass.js +0 -234
  228. package/lib/graphql/Schema.js +0 -197
  229. package/lib/graphql/index.js +0 -1
  230. package/lib/graphql/types/ACL.js +0 -35
  231. package/lib/graphql/types/Date.js +0 -25
  232. package/lib/graphql/types/File.js +0 -24
  233. package/lib/graphql/types/GeoPoint.js +0 -35
  234. package/lib/graphql/types/JSONObject.js +0 -30
  235. package/lib/graphql/types/NumberInput.js +0 -43
  236. package/lib/graphql/types/NumberQuery.js +0 -42
  237. package/lib/graphql/types/Pointer.js +0 -35
  238. package/lib/graphql/types/QueryConstraint.js +0 -61
  239. package/lib/graphql/types/StringQuery.js +0 -39
  240. package/lib/graphql/types/index.js +0 -110
@@ -1,206 +1,301 @@
1
- 'use strict';
1
+ "use strict";
2
2
 
3
3
  Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
- exports.FilesRouter = undefined;
7
-
8
- var _express = require('express');
9
-
10
- var _express2 = _interopRequireDefault(_express);
11
-
12
- var _bodyParser = require('body-parser');
13
-
14
- var _bodyParser2 = _interopRequireDefault(_bodyParser);
15
-
16
- var _middlewares = require('../middlewares');
17
-
18
- var Middlewares = _interopRequireWildcard(_middlewares);
19
-
20
- var _node = require('parse/node');
21
-
22
- var _node2 = _interopRequireDefault(_node);
23
-
24
- var _Config = require('../Config');
25
-
26
- var _Config2 = _interopRequireDefault(_Config);
27
-
28
- var _mime = require('mime');
29
-
30
- var _mime2 = _interopRequireDefault(_mime);
31
-
32
- var _logger = require('../logger');
33
-
34
- var _logger2 = _interopRequireDefault(_logger);
35
-
36
- function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } else { var newObj = {}; if (obj != null) { for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) newObj[key] = obj[key]; } } newObj.default = obj; return newObj; } }
37
-
38
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
39
-
6
+ exports.FilesRouter = void 0;
7
+ var _express = _interopRequireDefault(require("express"));
8
+ var Middlewares = _interopRequireWildcard(require("../middlewares"));
9
+ var _node = _interopRequireDefault(require("parse/node"));
10
+ var _Config = _interopRequireDefault(require("../Config"));
11
+ var _logger = _interopRequireDefault(require("../logger"));
12
+ 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
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
14
+ const triggers = require('../triggers');
15
+ const Utils = require('../Utils');
40
16
  class FilesRouter {
41
-
42
- expressRouter({ maxUploadSize = '20Mb' } = {}) {
43
- var router = _express2.default.Router();
17
+ expressRouter({
18
+ maxUploadSize = '20Mb'
19
+ } = {}) {
20
+ var router = _express.default.Router();
44
21
  router.get('/files/:appId/:filename', this.getHandler);
45
-
22
+ router.get('/files/:appId/metadata/:filename', this.metadataHandler);
46
23
  router.post('/files', function (req, res, next) {
47
- next(new _node2.default.Error(_node2.default.Error.INVALID_FILE_NAME, 'Filename not provided.'));
24
+ next(new _node.default.Error(_node.default.Error.INVALID_FILE_NAME, 'Filename not provided.'));
48
25
  });
49
-
50
- router.post('/files/:filename', Middlewares.allowCrossDomain, _bodyParser2.default.raw({ type: () => {
26
+ router.post('/files/:filename', _express.default.raw({
27
+ type: () => {
51
28
  return true;
52
- }, limit: maxUploadSize }), // Allow uploads without Content-Type, or with any Content-Type.
53
- Middlewares.handleParseHeaders, this.createHandler);
54
-
55
- router.delete('/files/:filename', Middlewares.allowCrossDomain, Middlewares.handleParseHeaders, Middlewares.enforceMasterKeyAccess, this.deleteHandler);
29
+ },
30
+ limit: maxUploadSize
31
+ }),
32
+ // Allow uploads without Content-Type, or with any Content-Type.
33
+ Middlewares.handleParseHeaders, Middlewares.handleParseSession, this.createHandler);
34
+ router.delete('/files/:filename', Middlewares.handleParseHeaders, Middlewares.handleParseSession, Middlewares.enforceMasterKeyAccess, this.deleteHandler);
56
35
  return router;
57
36
  }
58
-
59
- getHandler(req, res) {
60
- const config = _Config2.default.get(req.params.appId);
61
- const filesController = config.filesController;
62
- const filename = req.params.filename;
63
- const contentType = _mime2.default.getType(filename);
64
- if (isFileStreamable(req, filesController)) {
65
- filesController.getFileStream(config, filename).then(stream => {
66
- handleFileStream(stream, req, res, contentType);
67
- }).catch(() => {
68
- res.status(404);
69
- res.set('Content-Type', 'text/plain');
70
- res.end('File not found.');
37
+ async getHandler(req, res) {
38
+ const config = _Config.default.get(req.params.appId);
39
+ if (!config) {
40
+ res.status(403);
41
+ res.json({
42
+ code: _node.default.Error.OPERATION_FORBIDDEN,
43
+ error: 'Invalid application ID.'
71
44
  });
72
- } else {
73
- filesController.getFileData(config, filename).then(data => {
74
- res.status(200);
75
- res.set('Content-Type', contentType);
76
- res.set('Content-Length', data.length);
77
- res.end(data);
78
- }).catch(() => {
45
+ return;
46
+ }
47
+ let filename = req.params.filename;
48
+ try {
49
+ const filesController = config.filesController;
50
+ const mime = (await import('mime')).default;
51
+ let contentType = mime.getType(filename);
52
+ let file = new _node.default.File(filename, {
53
+ base64: ''
54
+ }, contentType);
55
+ const triggerResult = await triggers.maybeRunFileTrigger(triggers.Types.beforeFind, {
56
+ file
57
+ }, config, req.auth);
58
+ if (triggerResult?.file?._name) {
59
+ filename = triggerResult?.file?._name;
60
+ contentType = mime.getType(filename);
61
+ }
62
+ if (isFileStreamable(req, filesController)) {
63
+ filesController.handleFileStream(config, filename, req, res, contentType).catch(() => {
64
+ res.status(404);
65
+ res.set('Content-Type', 'text/plain');
66
+ res.end('File not found.');
67
+ });
68
+ return;
69
+ }
70
+ let data = await filesController.getFileData(config, filename).catch(() => {
79
71
  res.status(404);
80
72
  res.set('Content-Type', 'text/plain');
81
73
  res.end('File not found.');
82
74
  });
75
+ if (!data) {
76
+ return;
77
+ }
78
+ file = new _node.default.File(filename, {
79
+ base64: data.toString('base64')
80
+ }, contentType);
81
+ const afterFind = await triggers.maybeRunFileTrigger(triggers.Types.afterFind, {
82
+ file,
83
+ forceDownload: false
84
+ }, config, req.auth);
85
+ if (afterFind?.file) {
86
+ contentType = mime.getType(afterFind.file._name);
87
+ data = Buffer.from(afterFind.file._data, 'base64');
88
+ }
89
+ res.status(200);
90
+ res.set('Content-Type', contentType);
91
+ res.set('Content-Length', data.length);
92
+ if (afterFind.forceDownload) {
93
+ res.set('Content-Disposition', `attachment;filename=${afterFind.file._name}`);
94
+ }
95
+ res.end(data);
96
+ } catch (e) {
97
+ const err = triggers.resolveError(e, {
98
+ code: _node.default.Error.SCRIPT_FAILED,
99
+ message: `Could not find file: ${filename}.`
100
+ });
101
+ res.status(403);
102
+ res.json({
103
+ code: err.code,
104
+ error: err.message
105
+ });
83
106
  }
84
107
  }
85
-
86
- createHandler(req, res, next) {
87
- if (!req.body || !req.body.length) {
88
- next(new _node2.default.Error(_node2.default.Error.FILE_SAVE_ERROR, 'Invalid file upload.'));
108
+ async createHandler(req, res, next) {
109
+ const config = req.config;
110
+ const user = req.auth.user;
111
+ const isMaster = req.auth.isMaster;
112
+ const isLinked = user && _node.default.AnonymousUtils.isLinked(user);
113
+ if (!isMaster && !config.fileUpload.enableForAnonymousUser && isLinked) {
114
+ next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, 'File upload by anonymous user is disabled.'));
89
115
  return;
90
116
  }
91
-
92
- if (req.params.filename.length > 128) {
93
- next(new _node2.default.Error(_node2.default.Error.INVALID_FILE_NAME, 'Filename too long.'));
117
+ if (!isMaster && !config.fileUpload.enableForAuthenticatedUser && !isLinked && user) {
118
+ next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, 'File upload by authenticated user is disabled.'));
94
119
  return;
95
120
  }
96
-
97
- if (!req.params.filename.match(/^[_a-zA-Z0-9][a-zA-Z0-9@\.\ ~_-]*$/)) {
98
- next(new _node2.default.Error(_node2.default.Error.INVALID_FILE_NAME, 'Filename contains invalid characters.'));
121
+ if (!isMaster && !config.fileUpload.enableForPublic && !user) {
122
+ next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, 'File upload by public is disabled.'));
99
123
  return;
100
124
  }
101
-
102
- const filename = req.params.filename;
103
- const contentType = req.get('Content-type');
104
- const config = req.config;
105
125
  const filesController = config.filesController;
106
-
107
- filesController.createFile(config, filename, req.body, contentType).then(result => {
126
+ const {
127
+ filename
128
+ } = req.params;
129
+ 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
+ const error = filesController.validateFilename(filename);
135
+ if (error) {
136
+ next(error);
137
+ return;
138
+ }
139
+ const fileExtensions = config.fileUpload?.fileExtensions;
140
+ if (!isMaster && fileExtensions) {
141
+ const isValidExtension = extension => {
142
+ return fileExtensions.some(ext => {
143
+ if (ext === '*') {
144
+ return true;
145
+ }
146
+ const regex = new RegExp(ext);
147
+ if (regex.test(extension)) {
148
+ return true;
149
+ }
150
+ });
151
+ };
152
+ let extension = contentType;
153
+ if (filename && filename.includes('.')) {
154
+ extension = filename.substring(filename.lastIndexOf('.') + 1);
155
+ } else if (contentType && contentType.includes('/')) {
156
+ extension = contentType.split('/')[1];
157
+ }
158
+ extension = extension?.split(' ')?.join('');
159
+ if (extension && !isValidExtension(extension)) {
160
+ next(new _node.default.Error(_node.default.Error.FILE_SAVE_ERROR, `File upload of extension ${extension} is disabled.`));
161
+ return;
162
+ }
163
+ }
164
+ const base64 = req.body.toString('base64');
165
+ const file = new _node.default.File(filename, {
166
+ base64
167
+ }, contentType);
168
+ const {
169
+ metadata = {},
170
+ tags = {}
171
+ } = req.fileData || {};
172
+ try {
173
+ // Scan request data for denied keywords
174
+ Utils.checkProhibitedKeywords(config, metadata);
175
+ Utils.checkProhibitedKeywords(config, tags);
176
+ } catch (error) {
177
+ next(new _node.default.Error(_node.default.Error.INVALID_KEY_NAME, error));
178
+ return;
179
+ }
180
+ file.setTags(tags);
181
+ file.setMetadata(metadata);
182
+ const fileSize = Buffer.byteLength(req.body);
183
+ const fileObject = {
184
+ file,
185
+ fileSize
186
+ };
187
+ try {
188
+ // run beforeSaveFile trigger
189
+ const triggerResult = await triggers.maybeRunFileTrigger(triggers.Types.beforeSave, fileObject, config, req.auth);
190
+ let saveResult;
191
+ // if a new ParseFile is returned check if it's an already saved file
192
+ if (triggerResult instanceof _node.default.File) {
193
+ fileObject.file = triggerResult;
194
+ if (triggerResult.url()) {
195
+ // set fileSize to null because we wont know how big it is here
196
+ fileObject.fileSize = null;
197
+ saveResult = {
198
+ url: triggerResult.url(),
199
+ name: triggerResult._name
200
+ };
201
+ }
202
+ }
203
+ // if the file returned by the trigger has already been saved skip saving anything
204
+ if (!saveResult) {
205
+ // update fileSize
206
+ const bufferData = Buffer.from(fileObject.file._data, 'base64');
207
+ fileObject.fileSize = Buffer.byteLength(bufferData);
208
+ // prepare file options
209
+ const fileOptions = {
210
+ metadata: fileObject.file._metadata
211
+ };
212
+ // some s3-compatible providers (DigitalOcean, Linode) do not accept tags
213
+ // so we do not include the tags option if it is empty.
214
+ const fileTags = Object.keys(fileObject.file._tags).length > 0 ? {
215
+ tags: fileObject.file._tags
216
+ } : {};
217
+ Object.assign(fileOptions, fileTags);
218
+ // save file
219
+ const createFileResult = await filesController.createFile(config, fileObject.file._name, bufferData, fileObject.file._source.type, fileOptions);
220
+ // update file with new data
221
+ fileObject.file._name = createFileResult.name;
222
+ fileObject.file._url = createFileResult.url;
223
+ fileObject.file._requestTask = null;
224
+ fileObject.file._previousSave = Promise.resolve(fileObject.file);
225
+ saveResult = {
226
+ url: createFileResult.url,
227
+ name: createFileResult.name
228
+ };
229
+ }
230
+ // run afterSaveFile trigger
231
+ await triggers.maybeRunFileTrigger(triggers.Types.afterSave, fileObject, config, req.auth);
108
232
  res.status(201);
109
- res.set('Location', result.url);
110
- res.json(result);
111
- }).catch(e => {
112
- _logger2.default.error(e.message, e);
113
- next(new _node2.default.Error(_node2.default.Error.FILE_SAVE_ERROR, 'Could not store file.'));
114
- });
233
+ res.set('Location', saveResult.url);
234
+ res.json(saveResult);
235
+ } catch (e) {
236
+ _logger.default.error('Error creating a file: ', e);
237
+ const error = triggers.resolveError(e, {
238
+ code: _node.default.Error.FILE_SAVE_ERROR,
239
+ message: `Could not store file: ${fileObject.file._name}.`
240
+ });
241
+ next(error);
242
+ }
115
243
  }
116
-
117
- deleteHandler(req, res, next) {
118
- const filesController = req.config.filesController;
119
- filesController.deleteFile(req.config, req.params.filename).then(() => {
244
+ async deleteHandler(req, res, next) {
245
+ try {
246
+ const {
247
+ filesController
248
+ } = req.config;
249
+ const {
250
+ filename
251
+ } = req.params;
252
+ // run beforeDeleteFile trigger
253
+ const file = new _node.default.File(filename);
254
+ file._url = await filesController.adapter.getFileLocation(req.config, filename);
255
+ const fileObject = {
256
+ file,
257
+ fileSize: null
258
+ };
259
+ await triggers.maybeRunFileTrigger(triggers.Types.beforeDelete, fileObject, req.config, req.auth);
260
+ // delete file
261
+ await filesController.deleteFile(req.config, filename);
262
+ // run afterDeleteFile trigger
263
+ await triggers.maybeRunFileTrigger(triggers.Types.afterDelete, fileObject, req.config, req.auth);
120
264
  res.status(200);
121
265
  // TODO: return useful JSON here?
122
266
  res.end();
123
- }).catch(() => {
124
- next(new _node2.default.Error(_node2.default.Error.FILE_DELETE_ERROR, 'Could not delete file.'));
125
- });
267
+ } catch (e) {
268
+ _logger.default.error('Error deleting a file: ', e);
269
+ const error = triggers.resolveError(e, {
270
+ code: _node.default.Error.FILE_DELETE_ERROR,
271
+ message: 'Could not delete file.'
272
+ });
273
+ next(error);
274
+ }
275
+ }
276
+ async metadataHandler(req, res) {
277
+ try {
278
+ const config = _Config.default.get(req.params.appId);
279
+ const {
280
+ filesController
281
+ } = config;
282
+ const {
283
+ filename
284
+ } = req.params;
285
+ const data = await filesController.getMetadata(filename);
286
+ res.status(200);
287
+ res.json(data);
288
+ } catch {
289
+ res.status(200);
290
+ res.json({});
291
+ }
126
292
  }
127
293
  }
128
-
129
294
  exports.FilesRouter = FilesRouter;
130
295
  function isFileStreamable(req, filesController) {
131
- return req.get('Range') && typeof filesController.adapter.getFileStream === 'function';
132
- }
133
-
134
- function getRange(req) {
135
- const parts = req.get('Range').replace(/bytes=/, "").split("-");
136
- return { start: parseInt(parts[0], 10), end: parseInt(parts[1], 10) };
137
- }
138
-
139
- // handleFileStream is licenced under Creative Commons Attribution 4.0 International License (https://creativecommons.org/licenses/by/4.0/).
140
- // Author: LEROIB at weightingformypizza (https://weightingformypizza.wordpress.com/2015/06/24/stream-html5-media-content-like-video-audio-from-mongodb-using-express-and-gridstore/).
141
- function handleFileStream(stream, req, res, contentType) {
142
- const buffer_size = 1024 * 1024; //1024Kb
143
- // Range request, partiall stream the file
144
- let {
145
- start, end
146
- } = getRange(req);
147
-
148
- const notEnded = !end && end !== 0;
149
- const notStarted = !start && start !== 0;
150
- // No end provided, we want all bytes
151
- if (notEnded) {
152
- end = stream.length - 1;
153
- }
154
- // No start provided, we're reading backwards
155
- if (notStarted) {
156
- start = stream.length - end;
157
- end = start + end - 1;
158
- }
159
-
160
- // Data exceeds the buffer_size, cap
161
- if (end - start >= buffer_size) {
162
- end = start + buffer_size - 1;
163
- }
164
-
165
- const contentLength = end - start + 1;
166
-
167
- res.writeHead(206, {
168
- 'Content-Range': 'bytes ' + start + '-' + end + '/' + stream.length,
169
- 'Accept-Ranges': 'bytes',
170
- 'Content-Length': contentLength,
171
- 'Content-Type': contentType
172
- });
173
-
174
- stream.seek(start, function () {
175
- // get gridFile stream
176
- const gridFileStream = stream.stream(true);
177
- let bufferAvail = 0;
178
- let remainingBytesToWrite = contentLength;
179
- let totalBytesWritten = 0;
180
- // write to response
181
- gridFileStream.on('data', function (data) {
182
- bufferAvail += data.length;
183
- if (bufferAvail > 0) {
184
- // slice returns the same buffer if overflowing
185
- // safe to call in any case
186
- const buffer = data.slice(0, remainingBytesToWrite);
187
- // write the buffer
188
- res.write(buffer);
189
- // increment total
190
- totalBytesWritten += buffer.length;
191
- // decrement remaining
192
- remainingBytesToWrite -= data.length;
193
- // decrement the avaialbe buffer
194
- bufferAvail -= buffer.length;
195
- }
196
- // in case of small slices, all values will be good at that point
197
- // we've written enough, end...
198
- if (totalBytesWritten >= contentLength) {
199
- stream.close();
200
- res.end();
201
- this.destroy();
202
- }
203
- });
204
- });
296
+ const range = (req.get('Range') || '/-/').split('-');
297
+ const start = Number(range[0]);
298
+ const end = Number(range[1]);
299
+ return (!isNaN(start) || !isNaN(end)) && typeof filesController.adapter.handleFileStream === 'function';
205
300
  }
206
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Routers/FilesRouter.js"],"names":["Middlewares","FilesRouter","expressRouter","maxUploadSize","router","express","Router","get","getHandler","post","req","res","next","Parse","Error","INVALID_FILE_NAME","allowCrossDomain","BodyParser","raw","type","limit","handleParseHeaders","createHandler","delete","enforceMasterKeyAccess","deleteHandler","config","Config","params","appId","filesController","filename","contentType","mime","getType","isFileStreamable","getFileStream","then","stream","handleFileStream","catch","status","set","end","getFileData","data","length","body","FILE_SAVE_ERROR","match","createFile","result","url","json","e","logger","error","message","deleteFile","FILE_DELETE_ERROR","adapter","getRange","parts","replace","split","start","parseInt","buffer_size","notEnded","notStarted","contentLength","writeHead","seek","gridFileStream","bufferAvail","remainingBytesToWrite","totalBytesWritten","on","buffer","slice","write","close","destroy"],"mappings":"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","file":"FilesRouter.js","sourcesContent":["import express             from 'express';\nimport BodyParser          from 'body-parser';\nimport * as Middlewares    from '../middlewares';\nimport Parse               from 'parse/node';\nimport Config              from '../Config';\nimport mime                from 'mime';\nimport logger              from '../logger';\n\nexport class FilesRouter {\n\n  expressRouter({ maxUploadSize = '20Mb' } = {}) {\n    var router = express.Router();\n    router.get('/files/:appId/:filename', this.getHandler);\n\n    router.post('/files', function(req, res, next) {\n      next(new Parse.Error(Parse.Error.INVALID_FILE_NAME,\n        'Filename not provided.'));\n    });\n\n    router.post('/files/:filename',\n      Middlewares.allowCrossDomain,\n      BodyParser.raw({type: () => { return true; }, limit: maxUploadSize }), // Allow uploads without Content-Type, or with any Content-Type.\n      Middlewares.handleParseHeaders,\n      this.createHandler\n    );\n\n    router.delete('/files/:filename',\n      Middlewares.allowCrossDomain,\n      Middlewares.handleParseHeaders,\n      Middlewares.enforceMasterKeyAccess,\n      this.deleteHandler\n    );\n    return router;\n  }\n\n  getHandler(req, res) {\n    const config = Config.get(req.params.appId);\n    const filesController = config.filesController;\n    const filename = req.params.filename;\n    const contentType = mime.getType(filename);\n    if (isFileStreamable(req, filesController)) {\n      filesController.getFileStream(config, filename).then((stream) => {\n        handleFileStream(stream, req, res, contentType);\n      }).catch(() => {\n        res.status(404);\n        res.set('Content-Type', 'text/plain');\n        res.end('File not found.');\n      });\n    } else {\n      filesController.getFileData(config, filename).then((data) => {\n        res.status(200);\n        res.set('Content-Type', contentType);\n        res.set('Content-Length', data.length);\n        res.end(data);\n      }).catch(() => {\n        res.status(404);\n        res.set('Content-Type', 'text/plain');\n        res.end('File not found.');\n      });\n    }\n  }\n\n  createHandler(req, res, next) {\n    if (!req.body || !req.body.length) {\n      next(new Parse.Error(Parse.Error.FILE_SAVE_ERROR,\n        'Invalid file upload.'));\n      return;\n    }\n\n    if (req.params.filename.length > 128) {\n      next(new Parse.Error(Parse.Error.INVALID_FILE_NAME,\n        'Filename too long.'));\n      return;\n    }\n\n    if (!req.params.filename.match(/^[_a-zA-Z0-9][a-zA-Z0-9@\\.\\ ~_-]*$/)) {\n      next(new Parse.Error(Parse.Error.INVALID_FILE_NAME,\n        'Filename contains invalid characters.'));\n      return;\n    }\n\n    const filename = req.params.filename;\n    const contentType = req.get('Content-type');\n    const config = req.config;\n    const filesController = config.filesController;\n\n    filesController.createFile(config, filename, req.body, contentType).then((result) => {\n      res.status(201);\n      res.set('Location', result.url);\n      res.json(result);\n    }).catch((e) => {\n      logger.error(e.message, e);\n      next(new Parse.Error(Parse.Error.FILE_SAVE_ERROR, 'Could not store file.'));\n    });\n  }\n\n  deleteHandler(req, res, next) {\n    const filesController = req.config.filesController;\n    filesController.deleteFile(req.config, req.params.filename).then(() => {\n      res.status(200);\n      // TODO: return useful JSON here?\n      res.end();\n    }).catch(() => {\n      next(new Parse.Error(Parse.Error.FILE_DELETE_ERROR,\n        'Could not delete file.'));\n    });\n  }\n}\n\nfunction isFileStreamable(req, filesController){\n  return  req.get('Range') && typeof filesController.adapter.getFileStream === 'function';\n}\n\nfunction getRange(req) {\n  const parts = req.get('Range').replace(/bytes=/, \"\").split(\"-\");\n  return { start: parseInt(parts[0], 10), end: parseInt(parts[1], 10) };\n}\n\n// handleFileStream is licenced under Creative Commons Attribution 4.0 International License (https://creativecommons.org/licenses/by/4.0/).\n// Author: LEROIB at weightingformypizza (https://weightingformypizza.wordpress.com/2015/06/24/stream-html5-media-content-like-video-audio-from-mongodb-using-express-and-gridstore/).\nfunction handleFileStream(stream, req, res, contentType) {\n  const buffer_size = 1024 * 1024; //1024Kb\n  // Range request, partiall stream the file\n  let {\n    start, end\n  } = getRange(req);\n\n  const notEnded = (!end && end !== 0);\n  const notStarted = (!start && start !== 0);\n  // No end provided, we want all bytes\n  if (notEnded) {\n    end = stream.length - 1;\n  }\n  // No start provided, we're reading backwards\n  if (notStarted) {\n    start = stream.length - end;\n    end = start + end - 1;\n  }\n\n  // Data exceeds the buffer_size, cap\n  if (end - start >= buffer_size) {\n    end = start + buffer_size - 1;\n  }\n\n  const contentLength = (end - start) + 1;\n\n  res.writeHead(206, {\n    'Content-Range': 'bytes ' + start + '-' + end + '/' + stream.length,\n    'Accept-Ranges': 'bytes',\n    'Content-Length': contentLength,\n    'Content-Type': contentType,\n  });\n\n  stream.seek(start, function () {\n    // get gridFile stream\n    const gridFileStream = stream.stream(true);\n    let bufferAvail = 0;\n    let remainingBytesToWrite = contentLength;\n    let totalBytesWritten = 0;\n    // write to response\n    gridFileStream.on('data', function (data) {\n      bufferAvail += data.length;\n      if (bufferAvail > 0) {\n        // slice returns the same buffer if overflowing\n        // safe to call in any case\n        const buffer = data.slice(0, remainingBytesToWrite);\n        // write the buffer\n        res.write(buffer);\n        // increment total\n        totalBytesWritten += buffer.length;\n        // decrement remaining\n        remainingBytesToWrite -= data.length;\n        // decrement the avaialbe buffer\n        bufferAvail -= buffer.length;\n      }\n      // in case of small slices, all values will be good at that point\n      // we've written enough, end...\n      if (totalBytesWritten >= contentLength) {\n        stream.close();\n        res.end();\n        this.destroy();\n      }\n    });\n  });\n}\n"]}
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":"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","ignoreList":[]}