parse-server 6.0.0-alpha.8 → 6.0.0-beta.1

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 (192) hide show
  1. package/LICENSE +167 -25
  2. package/NOTICE +10 -0
  3. package/README.md +57 -33
  4. package/lib/AccountLockout.js +11 -26
  5. package/lib/Adapters/AdapterLoader.js +8 -14
  6. package/lib/Adapters/Analytics/AnalyticsAdapter.js +2 -8
  7. package/lib/Adapters/Auth/AuthAdapter.js +7 -16
  8. package/lib/Adapters/Auth/OAuth1Client.js +32 -57
  9. package/lib/Adapters/Auth/apple.js +6 -22
  10. package/lib/Adapters/Auth/facebook.js +7 -37
  11. package/lib/Adapters/Auth/gcenter.js +8 -37
  12. package/lib/Adapters/Auth/github.js +7 -10
  13. package/lib/Adapters/Auth/google.js +11 -34
  14. package/lib/Adapters/Auth/gpgames.js +5 -8
  15. package/lib/Adapters/Auth/httpsRequest.js +1 -7
  16. package/lib/Adapters/Auth/index.js +20 -65
  17. package/lib/Adapters/Auth/instagram.js +5 -9
  18. package/lib/Adapters/Auth/janraincapture.js +8 -12
  19. package/lib/Adapters/Auth/janrainengage.js +7 -11
  20. package/lib/Adapters/Auth/keycloak.js +5 -19
  21. package/lib/Adapters/Auth/ldap.js +1 -15
  22. package/lib/Adapters/Auth/line.js +7 -10
  23. package/lib/Adapters/Auth/linkedin.js +7 -12
  24. package/lib/Adapters/Auth/meetup.js +7 -10
  25. package/lib/Adapters/Auth/microsoft.js +7 -10
  26. package/lib/Adapters/Auth/oauth2.js +6 -18
  27. package/lib/Adapters/Auth/phantauth.js +8 -10
  28. package/lib/Adapters/Auth/qq.js +7 -13
  29. package/lib/Adapters/Auth/spotify.js +7 -14
  30. package/lib/Adapters/Auth/twitter.js +5 -15
  31. package/lib/Adapters/Auth/vkontakte.js +9 -15
  32. package/lib/Adapters/Auth/wechat.js +7 -10
  33. package/lib/Adapters/Auth/weibo.js +7 -11
  34. package/lib/Adapters/Cache/CacheAdapter.js +4 -12
  35. package/lib/Adapters/Cache/InMemoryCache.js +5 -19
  36. package/lib/Adapters/Cache/InMemoryCacheAdapter.js +1 -11
  37. package/lib/Adapters/Cache/LRUCache.js +1 -11
  38. package/lib/Adapters/Cache/NullCacheAdapter.js +1 -8
  39. package/lib/Adapters/Cache/RedisCacheAdapter.js +3 -30
  40. package/lib/Adapters/Cache/SchemaCache.js +1 -6
  41. package/lib/Adapters/Email/MailAdapter.js +2 -7
  42. package/lib/Adapters/Files/FilesAdapter.js +7 -21
  43. package/lib/Adapters/Files/GridFSBucketAdapter.js +6 -44
  44. package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
  45. package/lib/Adapters/Logger/LoggerAdapter.js +2 -11
  46. package/lib/Adapters/Logger/WinstonLogger.js +3 -30
  47. package/lib/Adapters/Logger/WinstonLoggerAdapter.js +5 -16
  48. package/lib/Adapters/MessageQueue/EventEmitterMQ.js +3 -20
  49. package/lib/Adapters/PubSub/EventEmitterPubSub.js +1 -16
  50. package/lib/Adapters/PubSub/PubSubAdapter.js +2 -9
  51. package/lib/Adapters/PubSub/RedisPubSub.js +12 -7
  52. package/lib/Adapters/Push/PushAdapter.js +2 -8
  53. package/lib/Adapters/Storage/Mongo/MongoCollection.js +12 -37
  54. package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +26 -79
  55. package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +78 -209
  56. package/lib/Adapters/Storage/Mongo/MongoTransform.js +82 -371
  57. package/lib/Adapters/Storage/Postgres/PostgresClient.js +1 -13
  58. package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +1 -20
  59. package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +119 -446
  60. package/lib/Adapters/Storage/Postgres/sql/index.js +4 -7
  61. package/lib/Adapters/Storage/StorageAdapter.js +1 -1
  62. package/lib/Adapters/WebSocketServer/WSAdapter.js +3 -12
  63. package/lib/Adapters/WebSocketServer/WSSAdapter.js +7 -12
  64. package/lib/Auth.js +68 -121
  65. package/lib/ClientSDK.js +3 -11
  66. package/lib/Config.js +73 -115
  67. package/lib/Controllers/AdaptableController.js +6 -18
  68. package/lib/Controllers/AnalyticsController.js +1 -9
  69. package/lib/Controllers/CacheController.js +3 -23
  70. package/lib/Controllers/DatabaseController.js +171 -364
  71. package/lib/Controllers/FilesController.js +5 -34
  72. package/lib/Controllers/HooksController.js +1 -51
  73. package/lib/Controllers/LiveQueryController.js +4 -23
  74. package/lib/Controllers/LoggerController.js +15 -54
  75. package/lib/Controllers/ParseGraphQLController.js +49 -104
  76. package/lib/Controllers/PushController.js +20 -59
  77. package/lib/Controllers/SchemaController.js +162 -348
  78. package/lib/Controllers/UserController.js +17 -78
  79. package/lib/Controllers/index.js +19 -68
  80. package/lib/Controllers/types.js +1 -1
  81. package/lib/Deprecator/Deprecations.js +1 -4
  82. package/lib/Deprecator/Deprecator.js +9 -18
  83. package/lib/GraphQL/ParseGraphQLSchema.js +16 -100
  84. package/lib/GraphQL/ParseGraphQLServer.js +2 -29
  85. package/lib/GraphQL/helpers/objectsMutations.js +2 -12
  86. package/lib/GraphQL/helpers/objectsQueries.js +18 -76
  87. package/lib/GraphQL/loaders/defaultGraphQLMutations.js +1 -9
  88. package/lib/GraphQL/loaders/defaultGraphQLQueries.js +1 -8
  89. package/lib/GraphQL/loaders/defaultGraphQLTypes.js +9 -115
  90. package/lib/GraphQL/loaders/defaultRelaySchema.js +6 -18
  91. package/lib/GraphQL/loaders/filesMutations.js +2 -19
  92. package/lib/GraphQL/loaders/functionsMutations.js +6 -17
  93. package/lib/GraphQL/loaders/parseClassMutations.js +6 -44
  94. package/lib/GraphQL/loaders/parseClassQueries.js +1 -26
  95. package/lib/GraphQL/loaders/parseClassTypes.js +10 -64
  96. package/lib/GraphQL/loaders/schemaDirectives.js +1 -17
  97. package/lib/GraphQL/loaders/schemaMutations.js +1 -20
  98. package/lib/GraphQL/loaders/schemaQueries.js +1 -14
  99. package/lib/GraphQL/loaders/schemaTypes.js +2 -6
  100. package/lib/GraphQL/loaders/usersMutations.js +6 -28
  101. package/lib/GraphQL/loaders/usersQueries.js +4 -26
  102. package/lib/GraphQL/parseGraphQLUtils.js +6 -19
  103. package/lib/GraphQL/transformers/className.js +1 -4
  104. package/lib/GraphQL/transformers/constraintType.js +1 -20
  105. package/lib/GraphQL/transformers/inputType.js +1 -20
  106. package/lib/GraphQL/transformers/mutation.js +6 -51
  107. package/lib/GraphQL/transformers/outputType.js +1 -20
  108. package/lib/GraphQL/transformers/query.js +6 -42
  109. package/lib/GraphQL/transformers/schemaFields.js +7 -34
  110. package/lib/KeyPromiseQueue.js +1 -12
  111. package/lib/LiveQuery/Client.js +1 -25
  112. package/lib/LiveQuery/Id.js +1 -7
  113. package/lib/LiveQuery/ParseCloudCodePublisher.js +13 -19
  114. package/lib/LiveQuery/ParseLiveQueryServer.js +111 -307
  115. package/lib/LiveQuery/ParsePubSub.js +1 -12
  116. package/lib/LiveQuery/ParseWebSocketServer.js +4 -26
  117. package/lib/LiveQuery/QueryTools.js +14 -116
  118. package/lib/LiveQuery/RequestSchema.js +1 -1
  119. package/lib/LiveQuery/SessionTokenCache.js +1 -17
  120. package/lib/LiveQuery/Subscription.js +4 -18
  121. package/lib/LiveQuery/equalObjects.js +2 -14
  122. package/lib/Options/Definitions.js +89 -9
  123. package/lib/Options/docs.js +25 -3
  124. package/lib/Options/index.js +4 -12
  125. package/lib/Options/parsers.js +1 -18
  126. package/lib/Page.js +1 -9
  127. package/lib/ParseMessageQueue.js +1 -10
  128. package/lib/ParseServer.js +175 -207
  129. package/lib/ParseServerRESTController.js +6 -33
  130. package/lib/PromiseRouter.js +16 -50
  131. package/lib/Push/PushQueue.js +3 -15
  132. package/lib/Push/PushWorker.js +7 -32
  133. package/lib/Push/utils.js +9 -38
  134. package/lib/RestQuery.js +105 -242
  135. package/lib/RestWrite.js +224 -389
  136. package/lib/Routers/AggregateRouter.js +14 -51
  137. package/lib/Routers/AnalyticsRouter.js +2 -8
  138. package/lib/Routers/AudiencesRouter.js +1 -15
  139. package/lib/Routers/ClassesRouter.js +3 -53
  140. package/lib/Routers/CloudCodeRouter.js +1 -19
  141. package/lib/Routers/FeaturesRouter.js +1 -10
  142. package/lib/Routers/FilesRouter.js +29 -76
  143. package/lib/Routers/FunctionsRouter.js +5 -28
  144. package/lib/Routers/GlobalConfigRouter.js +4 -18
  145. package/lib/Routers/GraphQLRouter.js +1 -14
  146. package/lib/Routers/HooksRouter.js +1 -29
  147. package/lib/Routers/IAPValidationRouter.js +6 -29
  148. package/lib/Routers/InstallationsRouter.js +2 -12
  149. package/lib/Routers/LogsRouter.js +4 -16
  150. package/lib/Routers/PagesRouter.js +69 -129
  151. package/lib/Routers/PublicAPIRouter.js +3 -62
  152. package/lib/Routers/PurgeRouter.js +1 -15
  153. package/lib/Routers/PushRouter.js +2 -18
  154. package/lib/Routers/RolesRouter.js +1 -7
  155. package/lib/Routers/SchemasRouter.js +4 -34
  156. package/lib/Routers/SecurityRouter.js +1 -12
  157. package/lib/Routers/SessionsRouter.js +3 -19
  158. package/lib/Routers/UsersRouter.js +48 -135
  159. package/lib/SchemaMigrations/DefinedSchemas.js +56 -115
  160. package/lib/SchemaMigrations/Migrations.js +2 -8
  161. package/lib/Security/Check.js +8 -16
  162. package/lib/Security/CheckGroup.js +4 -11
  163. package/lib/Security/CheckGroups/CheckGroupDatabase.js +8 -18
  164. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +5 -15
  165. package/lib/Security/CheckGroups/CheckGroups.js +1 -4
  166. package/lib/Security/CheckRunner.js +22 -41
  167. package/lib/StatusHandler.js +12 -69
  168. package/lib/TestUtils.js +1 -6
  169. package/lib/Utils.js +27 -66
  170. package/lib/batch.js +17 -28
  171. package/lib/cache.js +1 -3
  172. package/lib/cli/definitions/parse-live-query-server.js +1 -3
  173. package/lib/cli/definitions/parse-server.js +1 -3
  174. package/lib/cli/parse-live-query-server.js +1 -6
  175. package/lib/cli/parse-server.js +11 -21
  176. package/lib/cli/utils/commander.js +13 -51
  177. package/lib/cli/utils/runner.js +1 -14
  178. package/lib/cloud-code/Parse.Cloud.js +71 -81
  179. package/lib/cryptoUtils.js +11 -19
  180. package/lib/defaults.js +2 -14
  181. package/lib/deprecated.js +1 -2
  182. package/lib/index.js +16 -34
  183. package/lib/logger.js +6 -13
  184. package/lib/middlewares.js +166 -148
  185. package/lib/password.js +6 -10
  186. package/lib/request.js +8 -42
  187. package/lib/requiredParameter.js +1 -3
  188. package/lib/rest.js +25 -47
  189. package/lib/triggers.js +54 -252
  190. package/lib/vendor/mongodbUrl.js +129 -310
  191. package/package.json +13 -10
  192. package/PATENTS +0 -37
@@ -3,60 +3,49 @@
3
3
  Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
- exports.handleParseHeaders = handleParseHeaders;
6
+ exports.addRateLimit = exports.DEFAULT_ALLOWED_HEADERS = void 0;
7
7
  exports.allowCrossDomain = allowCrossDomain;
8
8
  exports.allowMethodOverride = allowMethodOverride;
9
- exports.handleParseErrors = handleParseErrors;
10
9
  exports.enforceMasterKeyAccess = enforceMasterKeyAccess;
10
+ exports.handleParseErrors = handleParseErrors;
11
+ exports.handleParseHeaders = handleParseHeaders;
12
+ exports.handleParseSession = void 0;
11
13
  exports.promiseEnforceMasterKeyAccess = promiseEnforceMasterKeyAccess;
12
14
  exports.promiseEnsureIdempotency = promiseEnsureIdempotency;
13
- exports.DEFAULT_ALLOWED_HEADERS = void 0;
14
-
15
15
  var _cache = _interopRequireDefault(require("./cache"));
16
-
17
16
  var _node = _interopRequireDefault(require("parse/node"));
18
-
19
17
  var _Auth = _interopRequireDefault(require("./Auth"));
20
-
21
18
  var _Config = _interopRequireDefault(require("./Config"));
22
-
23
19
  var _ClientSDK = _interopRequireDefault(require("./ClientSDK"));
24
-
25
20
  var _logger = _interopRequireDefault(require("./logger"));
26
-
27
21
  var _rest = _interopRequireDefault(require("./rest"));
28
-
29
22
  var _MongoStorageAdapter = _interopRequireDefault(require("./Adapters/Storage/Mongo/MongoStorageAdapter"));
30
-
31
23
  var _PostgresStorageAdapter = _interopRequireDefault(require("./Adapters/Storage/Postgres/PostgresStorageAdapter"));
32
-
24
+ var _expressRateLimit = _interopRequireDefault(require("express-rate-limit"));
25
+ var _Definitions = require("./Options/Definitions");
26
+ var _pathToRegexp = _interopRequireDefault(require("path-to-regexp"));
33
27
  var _ipRangeCheck = _interopRequireDefault(require("ip-range-check"));
34
-
35
28
  function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
36
-
37
29
  const DEFAULT_ALLOWED_HEADERS = 'X-Parse-Master-Key, X-Parse-REST-API-Key, X-Parse-Javascript-Key, X-Parse-Application-Id, X-Parse-Client-Version, X-Parse-Session-Token, X-Requested-With, X-Parse-Revocable-Session, X-Parse-Request-Id, Content-Type, Pragma, Cache-Control';
38
30
  exports.DEFAULT_ALLOWED_HEADERS = DEFAULT_ALLOWED_HEADERS;
39
-
40
31
  const getMountForRequest = function (req) {
41
32
  const mountPathLength = req.originalUrl.length - req.url.length;
42
33
  const mountPath = req.originalUrl.slice(0, mountPathLength);
43
34
  return req.protocol + '://' + req.get('host') + mountPath;
44
- }; // Checks that the request is authorized for this app and checks user
35
+ };
36
+
37
+ // Checks that the request is authorized for this app and checks user
45
38
  // auth too.
46
39
  // The bodyparser should run before this middleware.
47
40
  // Adds info to the request:
48
41
  // req.config - the Config for this app
49
42
  // req.auth - the Auth for this request
50
-
51
-
52
43
  function handleParseHeaders(req, res, next) {
53
44
  var mount = getMountForRequest(req);
54
45
  let context = {};
55
-
56
46
  if (req.get('X-Parse-Cloud-Context') != null) {
57
47
  try {
58
48
  context = JSON.parse(req.get('X-Parse-Cloud-Context'));
59
-
60
49
  if (Object.prototype.toString.call(context) !== '[object Object]') {
61
50
  throw 'Context is not an object';
62
51
  }
@@ -64,11 +53,11 @@ function handleParseHeaders(req, res, next) {
64
53
  return malformedContext(req, res);
65
54
  }
66
55
  }
67
-
68
56
  var info = {
69
57
  appId: req.get('X-Parse-Application-Id'),
70
58
  sessionToken: req.get('X-Parse-Session-Token'),
71
59
  masterKey: req.get('X-Parse-Master-Key'),
60
+ maintenanceKey: req.get('X-Parse-Maintenance-Key'),
72
61
  installationId: req.get('X-Parse-Installation-Id'),
73
62
  clientKey: req.get('X-Parse-Client-Key'),
74
63
  javascriptKey: req.get('X-Parse-Javascript-Key'),
@@ -78,25 +67,20 @@ function handleParseHeaders(req, res, next) {
78
67
  context: context
79
68
  };
80
69
  var basicAuth = httpAuth(req);
81
-
82
70
  if (basicAuth) {
83
71
  var basicAuthAppId = basicAuth.appId;
84
-
85
72
  if (_cache.default.get(basicAuthAppId)) {
86
73
  info.appId = basicAuthAppId;
87
74
  info.masterKey = basicAuth.masterKey || info.masterKey;
88
75
  info.javascriptKey = basicAuth.javascriptKey || info.javascriptKey;
89
76
  }
90
77
  }
91
-
92
78
  if (req.body) {
93
79
  // Unity SDK sends a _noBody key which needs to be removed.
94
80
  // Unclear at this point if action needs to be taken.
95
81
  delete req.body._noBody;
96
82
  }
97
-
98
83
  var fileViaJSON = false;
99
-
100
84
  if (!info.appId || !_cache.default.get(info.appId)) {
101
85
  // See if we can find the app id on the body.
102
86
  if (req.body instanceof Buffer) {
@@ -110,48 +94,40 @@ function handleParseHeaders(req, res, next) {
110
94
  } catch (e) {
111
95
  return invalidRequest(req, res);
112
96
  }
113
-
114
97
  fileViaJSON = true;
115
98
  }
116
-
117
99
  if (req.body) {
118
100
  delete req.body._RevocableSession;
119
101
  }
120
-
121
102
  if (req.body && req.body._ApplicationId && _cache.default.get(req.body._ApplicationId) && (!info.masterKey || _cache.default.get(req.body._ApplicationId).masterKey === info.masterKey)) {
122
103
  info.appId = req.body._ApplicationId;
123
104
  info.javascriptKey = req.body._JavaScriptKey || '';
124
105
  delete req.body._ApplicationId;
125
- delete req.body._JavaScriptKey; // TODO: test that the REST API formats generated by the other
106
+ delete req.body._JavaScriptKey;
107
+ // TODO: test that the REST API formats generated by the other
126
108
  // SDKs are handled ok
127
-
128
109
  if (req.body._ClientVersion) {
129
110
  info.clientVersion = req.body._ClientVersion;
130
111
  delete req.body._ClientVersion;
131
112
  }
132
-
133
113
  if (req.body._InstallationId) {
134
114
  info.installationId = req.body._InstallationId;
135
115
  delete req.body._InstallationId;
136
116
  }
137
-
138
117
  if (req.body._SessionToken) {
139
118
  info.sessionToken = req.body._SessionToken;
140
119
  delete req.body._SessionToken;
141
120
  }
142
-
143
121
  if (req.body._MasterKey) {
144
122
  info.masterKey = req.body._MasterKey;
145
123
  delete req.body._MasterKey;
146
124
  }
147
-
148
125
  if (req.body._context) {
149
126
  if (req.body._context instanceof Object) {
150
127
  info.context = req.body._context;
151
128
  } else {
152
129
  try {
153
130
  info.context = JSON.parse(req.body._context);
154
-
155
131
  if (Object.prototype.toString.call(info.context) !== '[object Object]') {
156
132
  throw 'Context is not an object';
157
133
  }
@@ -159,10 +135,8 @@ function handleParseHeaders(req, res, next) {
159
135
  return malformedContext(req, res);
160
136
  }
161
137
  }
162
-
163
138
  delete req.body._context;
164
139
  }
165
-
166
140
  if (req.body._ContentType) {
167
141
  req.headers['content-type'] = req.body._ContentType;
168
142
  delete req.body._ContentType;
@@ -171,46 +145,64 @@ function handleParseHeaders(req, res, next) {
171
145
  return invalidRequest(req, res);
172
146
  }
173
147
  }
174
-
175
148
  if (info.sessionToken && typeof info.sessionToken !== 'string') {
176
149
  info.sessionToken = info.sessionToken.toString();
177
150
  }
178
-
179
151
  if (info.clientVersion) {
180
152
  info.clientSDK = _ClientSDK.default.fromString(info.clientVersion);
181
153
  }
182
-
183
154
  if (fileViaJSON) {
184
- req.fileData = req.body.fileData; // We need to repopulate req.body with a buffer
185
-
155
+ req.fileData = req.body.fileData;
156
+ // We need to repopulate req.body with a buffer
186
157
  var base64 = req.body.base64;
187
158
  req.body = Buffer.from(base64, 'base64');
188
159
  }
189
-
190
160
  const clientIp = getClientIp(req);
161
+ const config = _Config.default.get(info.appId, mount);
162
+ if (config.state && config.state !== 'ok') {
163
+ res.status(500);
164
+ res.json({
165
+ code: _node.default.Error.INTERNAL_SERVER_ERROR,
166
+ error: `Invalid server state: ${config.state}`
167
+ });
168
+ return;
169
+ }
191
170
  info.app = _cache.default.get(info.appId);
192
- req.config = _Config.default.get(info.appId, mount);
171
+ req.config = config;
193
172
  req.config.headers = req.headers || {};
194
173
  req.config.ip = clientIp;
195
174
  req.info = info;
175
+ const isMaintenance = req.config.maintenanceKey && info.maintenanceKey === req.config.maintenanceKey;
176
+ if (isMaintenance) {
177
+ var _req$config;
178
+ if ((0, _ipRangeCheck.default)(clientIp, req.config.maintenanceKeyIps || [])) {
179
+ req.auth = new _Auth.default.Auth({
180
+ config: req.config,
181
+ installationId: info.installationId,
182
+ isMaintenance: true
183
+ });
184
+ next();
185
+ return;
186
+ }
187
+ const log = ((_req$config = req.config) === null || _req$config === void 0 ? void 0 : _req$config.loggerController) || _logger.default;
188
+ log.error(`Request using maintenance key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'maintenanceKeyIps'.`);
189
+ }
196
190
  let isMaster = info.masterKey === req.config.masterKey;
197
-
198
191
  if (isMaster && !(0, _ipRangeCheck.default)(clientIp, req.config.masterKeyIps || [])) {
192
+ var _req$config2;
193
+ const log = ((_req$config2 = req.config) === null || _req$config2 === void 0 ? void 0 : _req$config2.loggerController) || _logger.default;
194
+ log.error(`Request using master key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'masterKeyIps'.`);
199
195
  isMaster = false;
200
196
  }
201
-
202
197
  if (isMaster) {
203
198
  req.auth = new _Auth.default.Auth({
204
199
  config: req.config,
205
200
  installationId: info.installationId,
206
201
  isMaster: true
207
202
  });
208
- next();
209
- return;
203
+ return handleRateLimit(req, res, next);
210
204
  }
211
-
212
205
  var isReadOnlyMaster = info.masterKey === req.config.readOnlyMasterKey;
213
-
214
206
  if (typeof req.config.readOnlyMasterKey != 'undefined' && req.config.readOnlyMasterKey && isReadOnlyMaster) {
215
207
  req.auth = new _Auth.default.Auth({
216
208
  config: req.config,
@@ -218,12 +210,11 @@ function handleParseHeaders(req, res, next) {
218
210
  isMaster: true,
219
211
  isReadOnly: true
220
212
  });
221
- next();
222
- return;
223
- } // Client keys are not required in parse-server, but if any have been configured in the server, validate them
224
- // to preserve original behavior.
225
-
213
+ return handleRateLimit(req, res, next);
214
+ }
226
215
 
216
+ // Client keys are not required in parse-server, but if any have been configured in the server, validate them
217
+ // to preserve original behavior.
227
218
  const keys = ['clientKey', 'javascriptKey', 'dotNetKey', 'restAPIKey'];
228
219
  const oneKeyConfigured = keys.some(function (key) {
229
220
  return req.config[key] !== undefined;
@@ -231,15 +222,12 @@ function handleParseHeaders(req, res, next) {
231
222
  const oneKeyMatches = keys.some(function (key) {
232
223
  return req.config[key] !== undefined && info[key] === req.config[key];
233
224
  });
234
-
235
225
  if (oneKeyConfigured && !oneKeyMatches) {
236
226
  return invalidRequest(req, res);
237
227
  }
238
-
239
228
  if (req.url == '/login') {
240
229
  delete info.sessionToken;
241
230
  }
242
-
243
231
  if (req.userFromJWT) {
244
232
  req.auth = new _Auth.default.Auth({
245
233
  config: req.config,
@@ -247,89 +235,96 @@ function handleParseHeaders(req, res, next) {
247
235
  isMaster: false,
248
236
  user: req.userFromJWT
249
237
  });
250
- next();
251
- return;
238
+ return handleRateLimit(req, res, next);
252
239
  }
253
-
254
240
  if (!info.sessionToken) {
255
241
  req.auth = new _Auth.default.Auth({
256
242
  config: req.config,
257
243
  installationId: info.installationId,
258
244
  isMaster: false
259
245
  });
260
- next();
246
+ }
247
+ handleRateLimit(req, res, next);
248
+ }
249
+ const handleRateLimit = async (req, res, next) => {
250
+ const rateLimits = req.config.rateLimits || [];
251
+ try {
252
+ await Promise.all(rateLimits.map(async limit => {
253
+ const pathExp = new RegExp(limit.path);
254
+ if (pathExp.test(req.url)) {
255
+ await limit.handler(req, res, err => {
256
+ if (err) {
257
+ if (err.code === _node.default.Error.CONNECTION_FAILED) {
258
+ throw err;
259
+ }
260
+ req.config.loggerController.error('An unknown error occured when attempting to apply the rate limiter: ', err);
261
+ }
262
+ });
263
+ }
264
+ }));
265
+ } catch (error) {
266
+ res.status(429);
267
+ res.json({
268
+ code: _node.default.Error.CONNECTION_FAILED,
269
+ error: error.message
270
+ });
261
271
  return;
262
272
  }
263
-
264
- return Promise.resolve().then(() => {
265
- // handle the upgradeToRevocableSession path on it's own
273
+ next();
274
+ };
275
+ const handleParseSession = async (req, res, next) => {
276
+ try {
277
+ const info = req.info;
278
+ if (req.auth) {
279
+ next();
280
+ return;
281
+ }
282
+ let requestAuth = null;
266
283
  if (info.sessionToken && req.url === '/upgradeToRevocableSession' && info.sessionToken.indexOf('r:') != 0) {
267
- return _Auth.default.getAuthForLegacySessionToken({
284
+ requestAuth = await _Auth.default.getAuthForLegacySessionToken({
268
285
  config: req.config,
269
286
  installationId: info.installationId,
270
287
  sessionToken: info.sessionToken
271
288
  });
272
289
  } else {
273
- return _Auth.default.getAuthForSessionToken({
290
+ requestAuth = await _Auth.default.getAuthForSessionToken({
274
291
  config: req.config,
275
292
  installationId: info.installationId,
276
293
  sessionToken: info.sessionToken
277
294
  });
278
295
  }
279
- }).then(auth => {
280
- if (auth) {
281
- req.auth = auth;
282
- next();
283
- }
284
- }).catch(error => {
296
+ req.auth = requestAuth;
297
+ next();
298
+ } catch (error) {
285
299
  if (error instanceof _node.default.Error) {
286
300
  next(error);
287
301
  return;
288
- } else {
289
- // TODO: Determine the correct error scenario.
290
- req.config.loggerController.error('error getting auth for sessionToken', error);
291
- throw new _node.default.Error(_node.default.Error.UNKNOWN_ERROR, error);
292
302
  }
293
- });
294
- }
295
-
296
- function getClientIp(req) {
297
- if (req.headers['x-forwarded-for']) {
298
- // try to get from x-forwared-for if it set (behind reverse proxy)
299
- return req.headers['x-forwarded-for'].split(',')[0];
300
- } else if (req.connection && req.connection.remoteAddress) {
301
- // no proxy, try getting from connection.remoteAddress
302
- return req.connection.remoteAddress;
303
- } else if (req.socket) {
304
- // try to get it from req.socket
305
- return req.socket.remoteAddress;
306
- } else if (req.connection && req.connection.socket) {
307
- // try to get it form the connection.socket
308
- return req.connection.socket.remoteAddress;
309
- } else {
310
- // if non above, fallback.
311
- return req.ip;
303
+ // TODO: Determine the correct error scenario.
304
+ req.config.loggerController.error('error getting auth for sessionToken', error);
305
+ throw new _node.default.Error(_node.default.Error.UNKNOWN_ERROR, error);
312
306
  }
307
+ };
308
+ exports.handleParseSession = handleParseSession;
309
+ function getClientIp(req) {
310
+ return req.ip;
313
311
  }
314
-
315
312
  function httpAuth(req) {
316
313
  if (!(req.req || req).headers.authorization) return;
317
314
  var header = (req.req || req).headers.authorization;
318
- var appId, masterKey, javascriptKey; // parse header
315
+ var appId, masterKey, javascriptKey;
319
316
 
317
+ // parse header
320
318
  var authPrefix = 'basic ';
321
319
  var match = header.toLowerCase().indexOf(authPrefix);
322
-
323
320
  if (match == 0) {
324
321
  var encodedAuth = header.substring(authPrefix.length, header.length);
325
322
  var credentials = decodeBase64(encodedAuth).split(':');
326
-
327
323
  if (credentials.length == 2) {
328
324
  appId = credentials[0];
329
325
  var key = credentials[1];
330
326
  var jsKeyPrefix = 'javascript-key=';
331
327
  var matchKey = key.indexOf(jsKeyPrefix);
332
-
333
328
  if (matchKey == 0) {
334
329
  javascriptKey = key.substring(jsKeyPrefix.length, key.length);
335
330
  } else {
@@ -337,34 +332,28 @@ function httpAuth(req) {
337
332
  }
338
333
  }
339
334
  }
340
-
341
335
  return {
342
336
  appId: appId,
343
337
  masterKey: masterKey,
344
338
  javascriptKey: javascriptKey
345
339
  };
346
340
  }
347
-
348
341
  function decodeBase64(str) {
349
342
  return Buffer.from(str, 'base64').toString();
350
343
  }
351
-
352
344
  function allowCrossDomain(appId) {
353
345
  return (req, res, next) => {
354
346
  const config = _Config.default.get(appId, getMountForRequest(req));
355
-
356
347
  let allowHeaders = DEFAULT_ALLOWED_HEADERS;
357
-
358
348
  if (config && config.allowHeaders) {
359
349
  allowHeaders += `, ${config.allowHeaders.join(', ')}`;
360
350
  }
361
-
362
351
  const allowOrigin = config && config.allowOrigin || '*';
363
352
  res.header('Access-Control-Allow-Origin', allowOrigin);
364
353
  res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');
365
354
  res.header('Access-Control-Allow-Headers', allowHeaders);
366
- res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id'); // intercept OPTIONS method
367
-
355
+ res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id');
356
+ // intercept OPTIONS method
368
357
  if ('OPTIONS' == req.method) {
369
358
  res.sendStatus(200);
370
359
  } else {
@@ -372,40 +361,32 @@ function allowCrossDomain(appId) {
372
361
  }
373
362
  };
374
363
  }
375
-
376
364
  function allowMethodOverride(req, res, next) {
377
365
  if (req.method === 'POST' && req.body._method) {
378
366
  req.originalMethod = req.method;
379
367
  req.method = req.body._method;
380
368
  delete req.body._method;
381
369
  }
382
-
383
370
  next();
384
371
  }
385
-
386
372
  function handleParseErrors(err, req, res, next) {
387
373
  const log = req.config && req.config.loggerController || _logger.default;
388
-
389
374
  if (err instanceof _node.default.Error) {
390
375
  if (req.config && req.config.enableExpressErrorHandler) {
391
376
  return next(err);
392
377
  }
393
-
394
- let httpStatus; // TODO: fill out this mapping
395
-
378
+ let httpStatus;
379
+ // TODO: fill out this mapping
396
380
  switch (err.code) {
397
381
  case _node.default.Error.INTERNAL_SERVER_ERROR:
398
382
  httpStatus = 500;
399
383
  break;
400
-
401
384
  case _node.default.Error.OBJECT_NOT_FOUND:
402
385
  httpStatus = 404;
403
386
  break;
404
-
405
387
  default:
406
388
  httpStatus = 400;
407
389
  }
408
-
409
390
  res.status(httpStatus);
410
391
  res.json({
411
392
  code: err.code,
@@ -417,7 +398,6 @@ function handleParseErrors(err, req, res, next) {
417
398
  res.json({
418
399
  error: err.message
419
400
  });
420
-
421
401
  if (!(process && process.env.TESTING)) {
422
402
  next(err);
423
403
  }
@@ -428,23 +408,19 @@ function handleParseErrors(err, req, res, next) {
428
408
  code: _node.default.Error.INTERNAL_SERVER_ERROR,
429
409
  message: 'Internal server error.'
430
410
  });
431
-
432
411
  if (!(process && process.env.TESTING)) {
433
412
  next(err);
434
413
  }
435
414
  }
436
415
  }
437
-
438
416
  function enforceMasterKeyAccess(req, res, next) {
439
417
  if (!req.auth.isMaster) {
440
418
  res.status(403);
441
419
  res.end('{"error":"unauthorized: master key is required"}');
442
420
  return;
443
421
  }
444
-
445
422
  next();
446
423
  }
447
-
448
424
  function promiseEnforceMasterKeyAccess(request) {
449
425
  if (!request.auth.isMaster) {
450
426
  const error = new Error();
@@ -452,56 +428,101 @@ function promiseEnforceMasterKeyAccess(request) {
452
428
  error.message = 'unauthorized: master key is required';
453
429
  throw error;
454
430
  }
455
-
456
431
  return Promise.resolve();
457
432
  }
433
+ const addRateLimit = (route, config) => {
434
+ if (typeof config === 'string') {
435
+ config = _Config.default.get(config);
436
+ }
437
+ for (const key in route) {
438
+ if (!_Definitions.RateLimitOptions[key]) {
439
+ throw `Invalid rate limit option "${key}"`;
440
+ }
441
+ }
442
+ if (!config.rateLimits) {
443
+ config.rateLimits = [];
444
+ }
445
+ config.rateLimits.push({
446
+ path: (0, _pathToRegexp.default)(route.requestPath),
447
+ handler: (0, _expressRateLimit.default)({
448
+ windowMs: route.requestTimeWindow,
449
+ max: route.requestCount,
450
+ message: route.errorResponseMessage || _Definitions.RateLimitOptions.errorResponseMessage.default,
451
+ handler: (request, response, next, options) => {
452
+ throw {
453
+ code: _node.default.Error.CONNECTION_FAILED,
454
+ message: options.message
455
+ };
456
+ },
457
+ skip: request => {
458
+ var _request$auth;
459
+ if (request.ip === '127.0.0.1' && !route.includeInternalRequests) {
460
+ return true;
461
+ }
462
+ if (route.includeMasterKey) {
463
+ return false;
464
+ }
465
+ if (route.requestMethods) {
466
+ if (Array.isArray(route.requestMethods)) {
467
+ if (!route.requestMethods.includes(request.method)) {
468
+ return true;
469
+ }
470
+ } else {
471
+ const regExp = new RegExp(route.requestMethods);
472
+ if (!regExp.test(request.method)) {
473
+ return true;
474
+ }
475
+ }
476
+ }
477
+ return (_request$auth = request.auth) === null || _request$auth === void 0 ? void 0 : _request$auth.isMaster;
478
+ },
479
+ keyGenerator: request => {
480
+ return request.config.ip;
481
+ }
482
+ })
483
+ });
484
+ _Config.default.put(config);
485
+ };
486
+
458
487
  /**
459
488
  * Deduplicates a request to ensure idempotency. Duplicates are determined by the request ID
460
489
  * in the request header. If a request has no request ID, it is executed anyway.
461
490
  * @param {*} req The request to evaluate.
462
491
  * @returns Promise<{}>
463
492
  */
464
-
465
-
493
+ exports.addRateLimit = addRateLimit;
466
494
  function promiseEnsureIdempotency(req) {
467
495
  // Enable feature only for MongoDB
468
496
  if (!(req.config.database.adapter instanceof _MongoStorageAdapter.default || req.config.database.adapter instanceof _PostgresStorageAdapter.default)) {
469
497
  return Promise.resolve();
470
- } // Get parameters
471
-
472
-
498
+ }
499
+ // Get parameters
473
500
  const config = req.config;
474
501
  const requestId = ((req || {}).headers || {})['x-parse-request-id'];
475
502
  const {
476
503
  paths,
477
504
  ttl
478
505
  } = config.idempotencyOptions;
479
-
480
506
  if (!requestId || !config.idempotencyOptions) {
481
507
  return Promise.resolve();
482
- } // Request path may contain trailing slashes, depending on the original request, so remove
508
+ }
509
+ // Request path may contain trailing slashes, depending on the original request, so remove
483
510
  // leading and trailing slashes to make it easier to specify paths in the configuration
484
-
485
-
486
- const reqPath = req.path.replace(/^\/|\/$/, ''); // Determine whether idempotency is enabled for current request path
487
-
511
+ const reqPath = req.path.replace(/^\/|\/$/, '');
512
+ // Determine whether idempotency is enabled for current request path
488
513
  let match = false;
489
-
490
514
  for (const path of paths) {
491
515
  // Assume one wants a path to always match from the beginning to prevent any mistakes
492
516
  const regex = new RegExp(path.charAt(0) === '^' ? path : '^' + path);
493
-
494
517
  if (reqPath.match(regex)) {
495
518
  match = true;
496
519
  break;
497
520
  }
498
521
  }
499
-
500
522
  if (!match) {
501
523
  return Promise.resolve();
502
- } // Try to store request
503
-
504
-
524
+ }
525
+ // Try to store request
505
526
  const expiryDate = new Date(new Date().setSeconds(new Date().getSeconds() + ttl));
506
527
  return _rest.default.create(config, _Auth.default.master(config), '_Idempotency', {
507
528
  reqId: requestId,
@@ -510,16 +531,13 @@ function promiseEnsureIdempotency(req) {
510
531
  if (e.code == _node.default.Error.DUPLICATE_VALUE) {
511
532
  throw new _node.default.Error(_node.default.Error.DUPLICATE_REQUEST, 'Duplicate request');
512
533
  }
513
-
514
534
  throw e;
515
535
  });
516
536
  }
517
-
518
537
  function invalidRequest(req, res) {
519
538
  res.status(403);
520
539
  res.end('{"error":"unauthorized"}');
521
540
  }
522
-
523
541
  function malformedContext(req, res) {
524
542
  res.status(400);
525
543
  res.json({
@@ -527,4 +545,4 @@ function malformedContext(req, res) {
527
545
  error: 'Invalid object for context.'
528
546
  });
529
547
  }
530
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/middlewares.js"],"names":["DEFAULT_ALLOWED_HEADERS","getMountForRequest","req","mountPathLength","originalUrl","length","url","mountPath","slice","protocol","get","handleParseHeaders","res","next","mount","context","JSON","parse","Object","prototype","toString","call","e","malformedContext","info","appId","sessionToken","masterKey","installationId","clientKey","javascriptKey","dotNetKey","restAPIKey","clientVersion","basicAuth","httpAuth","basicAuthAppId","AppCache","body","_noBody","fileViaJSON","Buffer","invalidRequest","_RevocableSession","_ApplicationId","_JavaScriptKey","_ClientVersion","_InstallationId","_SessionToken","_MasterKey","_context","_ContentType","headers","clientSDK","ClientSDK","fromString","fileData","base64","from","clientIp","getClientIp","app","config","Config","ip","isMaster","masterKeyIps","auth","Auth","isReadOnlyMaster","readOnlyMasterKey","isReadOnly","keys","oneKeyConfigured","some","key","undefined","oneKeyMatches","userFromJWT","user","Promise","resolve","then","indexOf","getAuthForLegacySessionToken","getAuthForSessionToken","catch","error","Parse","Error","loggerController","UNKNOWN_ERROR","split","connection","remoteAddress","socket","authorization","header","authPrefix","match","toLowerCase","encodedAuth","substring","credentials","decodeBase64","jsKeyPrefix","matchKey","str","allowCrossDomain","allowHeaders","join","allowOrigin","method","sendStatus","allowMethodOverride","_method","originalMethod","handleParseErrors","err","log","defaultLogger","enableExpressErrorHandler","httpStatus","code","INTERNAL_SERVER_ERROR","OBJECT_NOT_FOUND","status","json","message","process","env","TESTING","stack","enforceMasterKeyAccess","end","promiseEnforceMasterKeyAccess","request","promiseEnsureIdempotency","database","adapter","MongoStorageAdapter","PostgresStorageAdapter","requestId","paths","ttl","idempotencyOptions","reqPath","path","replace","regex","RegExp","charAt","expiryDate","Date","setSeconds","getSeconds","rest","create","master","reqId","expire","_encode","DUPLICATE_VALUE","DUPLICATE_REQUEST","INVALID_JSON"],"mappings":"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","sourcesContent":["import AppCache from './cache';\nimport Parse from 'parse/node';\nimport auth from './Auth';\nimport Config from './Config';\nimport ClientSDK from './ClientSDK';\nimport defaultLogger from './logger';\nimport rest from './rest';\nimport MongoStorageAdapter from './Adapters/Storage/Mongo/MongoStorageAdapter';\nimport PostgresStorageAdapter from './Adapters/Storage/Postgres/PostgresStorageAdapter';\nimport ipRangeCheck from 'ip-range-check';\n\nexport const DEFAULT_ALLOWED_HEADERS =\n  'X-Parse-Master-Key, X-Parse-REST-API-Key, X-Parse-Javascript-Key, X-Parse-Application-Id, X-Parse-Client-Version, X-Parse-Session-Token, X-Requested-With, X-Parse-Revocable-Session, X-Parse-Request-Id, Content-Type, Pragma, Cache-Control';\n\nconst getMountForRequest = function (req) {\n  const mountPathLength = req.originalUrl.length - req.url.length;\n  const mountPath = req.originalUrl.slice(0, mountPathLength);\n  return req.protocol + '://' + req.get('host') + mountPath;\n};\n\n// Checks that the request is authorized for this app and checks user\n// auth too.\n// The bodyparser should run before this middleware.\n// Adds info to the request:\n// req.config - the Config for this app\n// req.auth - the Auth for this request\nexport function handleParseHeaders(req, res, next) {\n  var mount = getMountForRequest(req);\n\n  let context = {};\n  if (req.get('X-Parse-Cloud-Context') != null) {\n    try {\n      context = JSON.parse(req.get('X-Parse-Cloud-Context'));\n      if (Object.prototype.toString.call(context) !== '[object Object]') {\n        throw 'Context is not an object';\n      }\n    } catch (e) {\n      return malformedContext(req, res);\n    }\n  }\n  var info = {\n    appId: req.get('X-Parse-Application-Id'),\n    sessionToken: req.get('X-Parse-Session-Token'),\n    masterKey: req.get('X-Parse-Master-Key'),\n    installationId: req.get('X-Parse-Installation-Id'),\n    clientKey: req.get('X-Parse-Client-Key'),\n    javascriptKey: req.get('X-Parse-Javascript-Key'),\n    dotNetKey: req.get('X-Parse-Windows-Key'),\n    restAPIKey: req.get('X-Parse-REST-API-Key'),\n    clientVersion: req.get('X-Parse-Client-Version'),\n    context: context,\n  };\n\n  var basicAuth = httpAuth(req);\n\n  if (basicAuth) {\n    var basicAuthAppId = basicAuth.appId;\n    if (AppCache.get(basicAuthAppId)) {\n      info.appId = basicAuthAppId;\n      info.masterKey = basicAuth.masterKey || info.masterKey;\n      info.javascriptKey = basicAuth.javascriptKey || info.javascriptKey;\n    }\n  }\n\n  if (req.body) {\n    // Unity SDK sends a _noBody key which needs to be removed.\n    // Unclear at this point if action needs to be taken.\n    delete req.body._noBody;\n  }\n\n  var fileViaJSON = false;\n\n  if (!info.appId || !AppCache.get(info.appId)) {\n    // See if we can find the app id on the body.\n    if (req.body instanceof Buffer) {\n      // The only chance to find the app id is if this is a file\n      // upload that actually is a JSON body. So try to parse it.\n      // https://github.com/parse-community/parse-server/issues/6589\n      // It is also possible that the client is trying to upload a file but forgot\n      // to provide x-parse-app-id in header and parse a binary file will fail\n      try {\n        req.body = JSON.parse(req.body);\n      } catch (e) {\n        return invalidRequest(req, res);\n      }\n      fileViaJSON = true;\n    }\n\n    if (req.body) {\n      delete req.body._RevocableSession;\n    }\n\n    if (\n      req.body &&\n      req.body._ApplicationId &&\n      AppCache.get(req.body._ApplicationId) &&\n      (!info.masterKey || AppCache.get(req.body._ApplicationId).masterKey === info.masterKey)\n    ) {\n      info.appId = req.body._ApplicationId;\n      info.javascriptKey = req.body._JavaScriptKey || '';\n      delete req.body._ApplicationId;\n      delete req.body._JavaScriptKey;\n      // TODO: test that the REST API formats generated by the other\n      // SDKs are handled ok\n      if (req.body._ClientVersion) {\n        info.clientVersion = req.body._ClientVersion;\n        delete req.body._ClientVersion;\n      }\n      if (req.body._InstallationId) {\n        info.installationId = req.body._InstallationId;\n        delete req.body._InstallationId;\n      }\n      if (req.body._SessionToken) {\n        info.sessionToken = req.body._SessionToken;\n        delete req.body._SessionToken;\n      }\n      if (req.body._MasterKey) {\n        info.masterKey = req.body._MasterKey;\n        delete req.body._MasterKey;\n      }\n      if (req.body._context) {\n        if (req.body._context instanceof Object) {\n          info.context = req.body._context;\n        } else {\n          try {\n            info.context = JSON.parse(req.body._context);\n            if (Object.prototype.toString.call(info.context) !== '[object Object]') {\n              throw 'Context is not an object';\n            }\n          } catch (e) {\n            return malformedContext(req, res);\n          }\n        }\n        delete req.body._context;\n      }\n      if (req.body._ContentType) {\n        req.headers['content-type'] = req.body._ContentType;\n        delete req.body._ContentType;\n      }\n    } else {\n      return invalidRequest(req, res);\n    }\n  }\n\n  if (info.sessionToken && typeof info.sessionToken !== 'string') {\n    info.sessionToken = info.sessionToken.toString();\n  }\n\n  if (info.clientVersion) {\n    info.clientSDK = ClientSDK.fromString(info.clientVersion);\n  }\n\n  if (fileViaJSON) {\n    req.fileData = req.body.fileData;\n    // We need to repopulate req.body with a buffer\n    var base64 = req.body.base64;\n    req.body = Buffer.from(base64, 'base64');\n  }\n\n  const clientIp = getClientIp(req);\n\n  info.app = AppCache.get(info.appId);\n  req.config = Config.get(info.appId, mount);\n  req.config.headers = req.headers || {};\n  req.config.ip = clientIp;\n  req.info = info;\n\n  let isMaster = info.masterKey === req.config.masterKey;\n  if (isMaster && !ipRangeCheck(clientIp, req.config.masterKeyIps || [])) {\n    isMaster = false;\n  }\n\n  if (isMaster) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: true,\n    });\n    next();\n    return;\n  }\n\n  var isReadOnlyMaster = info.masterKey === req.config.readOnlyMasterKey;\n  if (\n    typeof req.config.readOnlyMasterKey != 'undefined' &&\n    req.config.readOnlyMasterKey &&\n    isReadOnlyMaster\n  ) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: true,\n      isReadOnly: true,\n    });\n    next();\n    return;\n  }\n\n  // Client keys are not required in parse-server, but if any have been configured in the server, validate them\n  //  to preserve original behavior.\n  const keys = ['clientKey', 'javascriptKey', 'dotNetKey', 'restAPIKey'];\n  const oneKeyConfigured = keys.some(function (key) {\n    return req.config[key] !== undefined;\n  });\n  const oneKeyMatches = keys.some(function (key) {\n    return req.config[key] !== undefined && info[key] === req.config[key];\n  });\n\n  if (oneKeyConfigured && !oneKeyMatches) {\n    return invalidRequest(req, res);\n  }\n\n  if (req.url == '/login') {\n    delete info.sessionToken;\n  }\n\n  if (req.userFromJWT) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: false,\n      user: req.userFromJWT,\n    });\n    next();\n    return;\n  }\n\n  if (!info.sessionToken) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: false,\n    });\n    next();\n    return;\n  }\n\n  return Promise.resolve()\n    .then(() => {\n      // handle the upgradeToRevocableSession path on it's own\n      if (\n        info.sessionToken &&\n        req.url === '/upgradeToRevocableSession' &&\n        info.sessionToken.indexOf('r:') != 0\n      ) {\n        return auth.getAuthForLegacySessionToken({\n          config: req.config,\n          installationId: info.installationId,\n          sessionToken: info.sessionToken,\n        });\n      } else {\n        return auth.getAuthForSessionToken({\n          config: req.config,\n          installationId: info.installationId,\n          sessionToken: info.sessionToken,\n        });\n      }\n    })\n    .then(auth => {\n      if (auth) {\n        req.auth = auth;\n        next();\n      }\n    })\n    .catch(error => {\n      if (error instanceof Parse.Error) {\n        next(error);\n        return;\n      } else {\n        // TODO: Determine the correct error scenario.\n        req.config.loggerController.error('error getting auth for sessionToken', error);\n        throw new Parse.Error(Parse.Error.UNKNOWN_ERROR, error);\n      }\n    });\n}\n\nfunction getClientIp(req) {\n  if (req.headers['x-forwarded-for']) {\n    // try to get from x-forwared-for if it set (behind reverse proxy)\n    return req.headers['x-forwarded-for'].split(',')[0];\n  } else if (req.connection && req.connection.remoteAddress) {\n    // no proxy, try getting from connection.remoteAddress\n    return req.connection.remoteAddress;\n  } else if (req.socket) {\n    // try to get it from req.socket\n    return req.socket.remoteAddress;\n  } else if (req.connection && req.connection.socket) {\n    // try to get it form the connection.socket\n    return req.connection.socket.remoteAddress;\n  } else {\n    // if non above, fallback.\n    return req.ip;\n  }\n}\n\nfunction httpAuth(req) {\n  if (!(req.req || req).headers.authorization) return;\n\n  var header = (req.req || req).headers.authorization;\n  var appId, masterKey, javascriptKey;\n\n  // parse header\n  var authPrefix = 'basic ';\n\n  var match = header.toLowerCase().indexOf(authPrefix);\n\n  if (match == 0) {\n    var encodedAuth = header.substring(authPrefix.length, header.length);\n    var credentials = decodeBase64(encodedAuth).split(':');\n\n    if (credentials.length == 2) {\n      appId = credentials[0];\n      var key = credentials[1];\n\n      var jsKeyPrefix = 'javascript-key=';\n\n      var matchKey = key.indexOf(jsKeyPrefix);\n      if (matchKey == 0) {\n        javascriptKey = key.substring(jsKeyPrefix.length, key.length);\n      } else {\n        masterKey = key;\n      }\n    }\n  }\n\n  return { appId: appId, masterKey: masterKey, javascriptKey: javascriptKey };\n}\n\nfunction decodeBase64(str) {\n  return Buffer.from(str, 'base64').toString();\n}\n\nexport function allowCrossDomain(appId) {\n  return (req, res, next) => {\n    const config = Config.get(appId, getMountForRequest(req));\n    let allowHeaders = DEFAULT_ALLOWED_HEADERS;\n    if (config && config.allowHeaders) {\n      allowHeaders += `, ${config.allowHeaders.join(', ')}`;\n    }\n    const allowOrigin = (config && config.allowOrigin) || '*';\n    res.header('Access-Control-Allow-Origin', allowOrigin);\n    res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');\n    res.header('Access-Control-Allow-Headers', allowHeaders);\n    res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id');\n    // intercept OPTIONS method\n    if ('OPTIONS' == req.method) {\n      res.sendStatus(200);\n    } else {\n      next();\n    }\n  };\n}\n\nexport function allowMethodOverride(req, res, next) {\n  if (req.method === 'POST' && req.body._method) {\n    req.originalMethod = req.method;\n    req.method = req.body._method;\n    delete req.body._method;\n  }\n  next();\n}\n\nexport function handleParseErrors(err, req, res, next) {\n  const log = (req.config && req.config.loggerController) || defaultLogger;\n  if (err instanceof Parse.Error) {\n    if (req.config && req.config.enableExpressErrorHandler) {\n      return next(err);\n    }\n    let httpStatus;\n    // TODO: fill out this mapping\n    switch (err.code) {\n      case Parse.Error.INTERNAL_SERVER_ERROR:\n        httpStatus = 500;\n        break;\n      case Parse.Error.OBJECT_NOT_FOUND:\n        httpStatus = 404;\n        break;\n      default:\n        httpStatus = 400;\n    }\n    res.status(httpStatus);\n    res.json({ code: err.code, error: err.message });\n    log.error('Parse error: ', err);\n  } else if (err.status && err.message) {\n    res.status(err.status);\n    res.json({ error: err.message });\n    if (!(process && process.env.TESTING)) {\n      next(err);\n    }\n  } else {\n    log.error('Uncaught internal server error.', err, err.stack);\n    res.status(500);\n    res.json({\n      code: Parse.Error.INTERNAL_SERVER_ERROR,\n      message: 'Internal server error.',\n    });\n    if (!(process && process.env.TESTING)) {\n      next(err);\n    }\n  }\n}\n\nexport function enforceMasterKeyAccess(req, res, next) {\n  if (!req.auth.isMaster) {\n    res.status(403);\n    res.end('{\"error\":\"unauthorized: master key is required\"}');\n    return;\n  }\n  next();\n}\n\nexport function promiseEnforceMasterKeyAccess(request) {\n  if (!request.auth.isMaster) {\n    const error = new Error();\n    error.status = 403;\n    error.message = 'unauthorized: master key is required';\n    throw error;\n  }\n  return Promise.resolve();\n}\n\n/**\n * Deduplicates a request to ensure idempotency. Duplicates are determined by the request ID\n * in the request header. If a request has no request ID, it is executed anyway.\n * @param {*} req The request to evaluate.\n * @returns Promise<{}>\n */\nexport function promiseEnsureIdempotency(req) {\n  // Enable feature only for MongoDB\n  if (\n    !(\n      req.config.database.adapter instanceof MongoStorageAdapter ||\n      req.config.database.adapter instanceof PostgresStorageAdapter\n    )\n  ) {\n    return Promise.resolve();\n  }\n  // Get parameters\n  const config = req.config;\n  const requestId = ((req || {}).headers || {})['x-parse-request-id'];\n  const { paths, ttl } = config.idempotencyOptions;\n  if (!requestId || !config.idempotencyOptions) {\n    return Promise.resolve();\n  }\n  // Request path may contain trailing slashes, depending on the original request, so remove\n  // leading and trailing slashes to make it easier to specify paths in the configuration\n  const reqPath = req.path.replace(/^\\/|\\/$/, '');\n  // Determine whether idempotency is enabled for current request path\n  let match = false;\n  for (const path of paths) {\n    // Assume one wants a path to always match from the beginning to prevent any mistakes\n    const regex = new RegExp(path.charAt(0) === '^' ? path : '^' + path);\n    if (reqPath.match(regex)) {\n      match = true;\n      break;\n    }\n  }\n  if (!match) {\n    return Promise.resolve();\n  }\n  // Try to store request\n  const expiryDate = new Date(new Date().setSeconds(new Date().getSeconds() + ttl));\n  return rest\n    .create(config, auth.master(config), '_Idempotency', {\n      reqId: requestId,\n      expire: Parse._encode(expiryDate),\n    })\n    .catch(e => {\n      if (e.code == Parse.Error.DUPLICATE_VALUE) {\n        throw new Parse.Error(Parse.Error.DUPLICATE_REQUEST, 'Duplicate request');\n      }\n      throw e;\n    });\n}\n\nfunction invalidRequest(req, res) {\n  res.status(403);\n  res.end('{\"error\":\"unauthorized\"}');\n}\n\nfunction malformedContext(req, res) {\n  res.status(400);\n  res.json({ code: Parse.Error.INVALID_JSON, error: 'Invalid object for context.' });\n}\n"]}
548
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["DEFAULT_ALLOWED_HEADERS","getMountForRequest","req","mountPathLength","originalUrl","length","url","mountPath","slice","protocol","get","handleParseHeaders","res","next","mount","context","JSON","parse","Object","prototype","toString","call","e","malformedContext","info","appId","sessionToken","masterKey","maintenanceKey","installationId","clientKey","javascriptKey","dotNetKey","restAPIKey","clientVersion","basicAuth","httpAuth","basicAuthAppId","AppCache","body","_noBody","fileViaJSON","Buffer","invalidRequest","_RevocableSession","_ApplicationId","_JavaScriptKey","_ClientVersion","_InstallationId","_SessionToken","_MasterKey","_context","_ContentType","headers","clientSDK","ClientSDK","fromString","fileData","base64","from","clientIp","getClientIp","config","Config","state","status","json","code","Parse","Error","INTERNAL_SERVER_ERROR","error","app","ip","isMaintenance","ipRangeCheck","maintenanceKeyIps","auth","Auth","log","loggerController","defaultLogger","isMaster","masterKeyIps","handleRateLimit","isReadOnlyMaster","readOnlyMasterKey","isReadOnly","keys","oneKeyConfigured","some","key","undefined","oneKeyMatches","userFromJWT","user","rateLimits","Promise","all","map","limit","pathExp","RegExp","path","test","handler","err","CONNECTION_FAILED","message","handleParseSession","requestAuth","indexOf","getAuthForLegacySessionToken","getAuthForSessionToken","UNKNOWN_ERROR","authorization","header","authPrefix","match","toLowerCase","encodedAuth","substring","credentials","decodeBase64","split","jsKeyPrefix","matchKey","str","allowCrossDomain","allowHeaders","join","allowOrigin","method","sendStatus","allowMethodOverride","_method","originalMethod","handleParseErrors","enableExpressErrorHandler","httpStatus","OBJECT_NOT_FOUND","process","env","TESTING","stack","enforceMasterKeyAccess","end","promiseEnforceMasterKeyAccess","request","resolve","addRateLimit","route","RateLimitOptions","push","pathToRegexp","requestPath","rateLimit","windowMs","requestTimeWindow","max","requestCount","errorResponseMessage","default","response","options","skip","includeInternalRequests","includeMasterKey","requestMethods","Array","isArray","includes","regExp","keyGenerator","put","promiseEnsureIdempotency","database","adapter","MongoStorageAdapter","PostgresStorageAdapter","requestId","paths","ttl","idempotencyOptions","reqPath","replace","regex","charAt","expiryDate","Date","setSeconds","getSeconds","rest","create","master","reqId","expire","_encode","catch","DUPLICATE_VALUE","DUPLICATE_REQUEST","INVALID_JSON"],"sources":["../src/middlewares.js"],"sourcesContent":["import AppCache from './cache';\nimport Parse from 'parse/node';\nimport auth from './Auth';\nimport Config from './Config';\nimport ClientSDK from './ClientSDK';\nimport defaultLogger from './logger';\nimport rest from './rest';\nimport MongoStorageAdapter from './Adapters/Storage/Mongo/MongoStorageAdapter';\nimport PostgresStorageAdapter from './Adapters/Storage/Postgres/PostgresStorageAdapter';\nimport rateLimit from 'express-rate-limit';\nimport { RateLimitOptions } from './Options/Definitions';\nimport pathToRegexp from 'path-to-regexp';\nimport ipRangeCheck from 'ip-range-check';\n\nexport const DEFAULT_ALLOWED_HEADERS =\n  'X-Parse-Master-Key, X-Parse-REST-API-Key, X-Parse-Javascript-Key, X-Parse-Application-Id, X-Parse-Client-Version, X-Parse-Session-Token, X-Requested-With, X-Parse-Revocable-Session, X-Parse-Request-Id, Content-Type, Pragma, Cache-Control';\n\nconst getMountForRequest = function (req) {\n  const mountPathLength = req.originalUrl.length - req.url.length;\n  const mountPath = req.originalUrl.slice(0, mountPathLength);\n  return req.protocol + '://' + req.get('host') + mountPath;\n};\n\n// Checks that the request is authorized for this app and checks user\n// auth too.\n// The bodyparser should run before this middleware.\n// Adds info to the request:\n// req.config - the Config for this app\n// req.auth - the Auth for this request\nexport function handleParseHeaders(req, res, next) {\n  var mount = getMountForRequest(req);\n\n  let context = {};\n  if (req.get('X-Parse-Cloud-Context') != null) {\n    try {\n      context = JSON.parse(req.get('X-Parse-Cloud-Context'));\n      if (Object.prototype.toString.call(context) !== '[object Object]') {\n        throw 'Context is not an object';\n      }\n    } catch (e) {\n      return malformedContext(req, res);\n    }\n  }\n  var info = {\n    appId: req.get('X-Parse-Application-Id'),\n    sessionToken: req.get('X-Parse-Session-Token'),\n    masterKey: req.get('X-Parse-Master-Key'),\n    maintenanceKey: req.get('X-Parse-Maintenance-Key'),\n    installationId: req.get('X-Parse-Installation-Id'),\n    clientKey: req.get('X-Parse-Client-Key'),\n    javascriptKey: req.get('X-Parse-Javascript-Key'),\n    dotNetKey: req.get('X-Parse-Windows-Key'),\n    restAPIKey: req.get('X-Parse-REST-API-Key'),\n    clientVersion: req.get('X-Parse-Client-Version'),\n    context: context,\n  };\n\n  var basicAuth = httpAuth(req);\n\n  if (basicAuth) {\n    var basicAuthAppId = basicAuth.appId;\n    if (AppCache.get(basicAuthAppId)) {\n      info.appId = basicAuthAppId;\n      info.masterKey = basicAuth.masterKey || info.masterKey;\n      info.javascriptKey = basicAuth.javascriptKey || info.javascriptKey;\n    }\n  }\n\n  if (req.body) {\n    // Unity SDK sends a _noBody key which needs to be removed.\n    // Unclear at this point if action needs to be taken.\n    delete req.body._noBody;\n  }\n\n  var fileViaJSON = false;\n\n  if (!info.appId || !AppCache.get(info.appId)) {\n    // See if we can find the app id on the body.\n    if (req.body instanceof Buffer) {\n      // The only chance to find the app id is if this is a file\n      // upload that actually is a JSON body. So try to parse it.\n      // https://github.com/parse-community/parse-server/issues/6589\n      // It is also possible that the client is trying to upload a file but forgot\n      // to provide x-parse-app-id in header and parse a binary file will fail\n      try {\n        req.body = JSON.parse(req.body);\n      } catch (e) {\n        return invalidRequest(req, res);\n      }\n      fileViaJSON = true;\n    }\n\n    if (req.body) {\n      delete req.body._RevocableSession;\n    }\n\n    if (\n      req.body &&\n      req.body._ApplicationId &&\n      AppCache.get(req.body._ApplicationId) &&\n      (!info.masterKey || AppCache.get(req.body._ApplicationId).masterKey === info.masterKey)\n    ) {\n      info.appId = req.body._ApplicationId;\n      info.javascriptKey = req.body._JavaScriptKey || '';\n      delete req.body._ApplicationId;\n      delete req.body._JavaScriptKey;\n      // TODO: test that the REST API formats generated by the other\n      // SDKs are handled ok\n      if (req.body._ClientVersion) {\n        info.clientVersion = req.body._ClientVersion;\n        delete req.body._ClientVersion;\n      }\n      if (req.body._InstallationId) {\n        info.installationId = req.body._InstallationId;\n        delete req.body._InstallationId;\n      }\n      if (req.body._SessionToken) {\n        info.sessionToken = req.body._SessionToken;\n        delete req.body._SessionToken;\n      }\n      if (req.body._MasterKey) {\n        info.masterKey = req.body._MasterKey;\n        delete req.body._MasterKey;\n      }\n      if (req.body._context) {\n        if (req.body._context instanceof Object) {\n          info.context = req.body._context;\n        } else {\n          try {\n            info.context = JSON.parse(req.body._context);\n            if (Object.prototype.toString.call(info.context) !== '[object Object]') {\n              throw 'Context is not an object';\n            }\n          } catch (e) {\n            return malformedContext(req, res);\n          }\n        }\n        delete req.body._context;\n      }\n      if (req.body._ContentType) {\n        req.headers['content-type'] = req.body._ContentType;\n        delete req.body._ContentType;\n      }\n    } else {\n      return invalidRequest(req, res);\n    }\n  }\n\n  if (info.sessionToken && typeof info.sessionToken !== 'string') {\n    info.sessionToken = info.sessionToken.toString();\n  }\n\n  if (info.clientVersion) {\n    info.clientSDK = ClientSDK.fromString(info.clientVersion);\n  }\n\n  if (fileViaJSON) {\n    req.fileData = req.body.fileData;\n    // We need to repopulate req.body with a buffer\n    var base64 = req.body.base64;\n    req.body = Buffer.from(base64, 'base64');\n  }\n\n  const clientIp = getClientIp(req);\n  const config = Config.get(info.appId, mount);\n  if (config.state && config.state !== 'ok') {\n    res.status(500);\n    res.json({\n      code: Parse.Error.INTERNAL_SERVER_ERROR,\n      error: `Invalid server state: ${config.state}`,\n    });\n    return;\n  }\n\n  info.app = AppCache.get(info.appId);\n  req.config = config;\n  req.config.headers = req.headers || {};\n  req.config.ip = clientIp;\n  req.info = info;\n\n  const isMaintenance =\n    req.config.maintenanceKey && info.maintenanceKey === req.config.maintenanceKey;\n  if (isMaintenance) {\n    if (ipRangeCheck(clientIp, req.config.maintenanceKeyIps || [])) {\n      req.auth = new auth.Auth({\n        config: req.config,\n        installationId: info.installationId,\n        isMaintenance: true,\n      });\n      next();\n      return;\n    }\n    const log = req.config?.loggerController || defaultLogger;\n    log.error(\n      `Request using maintenance key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'maintenanceKeyIps'.`\n    );\n  }\n\n  let isMaster = info.masterKey === req.config.masterKey;\n  if (isMaster && !ipRangeCheck(clientIp, req.config.masterKeyIps || [])) {\n    const log = req.config?.loggerController || defaultLogger;\n    log.error(\n      `Request using master key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'masterKeyIps'.`\n    );\n    isMaster = false;\n  }\n\n  if (isMaster) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: true,\n    });\n    return handleRateLimit(req, res, next);\n  }\n\n  var isReadOnlyMaster = info.masterKey === req.config.readOnlyMasterKey;\n  if (\n    typeof req.config.readOnlyMasterKey != 'undefined' &&\n    req.config.readOnlyMasterKey &&\n    isReadOnlyMaster\n  ) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: true,\n      isReadOnly: true,\n    });\n    return handleRateLimit(req, res, next);\n  }\n\n  // Client keys are not required in parse-server, but if any have been configured in the server, validate them\n  //  to preserve original behavior.\n  const keys = ['clientKey', 'javascriptKey', 'dotNetKey', 'restAPIKey'];\n  const oneKeyConfigured = keys.some(function (key) {\n    return req.config[key] !== undefined;\n  });\n  const oneKeyMatches = keys.some(function (key) {\n    return req.config[key] !== undefined && info[key] === req.config[key];\n  });\n\n  if (oneKeyConfigured && !oneKeyMatches) {\n    return invalidRequest(req, res);\n  }\n\n  if (req.url == '/login') {\n    delete info.sessionToken;\n  }\n\n  if (req.userFromJWT) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: false,\n      user: req.userFromJWT,\n    });\n    return handleRateLimit(req, res, next);\n  }\n\n  if (!info.sessionToken) {\n    req.auth = new auth.Auth({\n      config: req.config,\n      installationId: info.installationId,\n      isMaster: false,\n    });\n  }\n  handleRateLimit(req, res, next);\n}\n\nconst handleRateLimit = async (req, res, next) => {\n  const rateLimits = req.config.rateLimits || [];\n  try {\n    await Promise.all(\n      rateLimits.map(async limit => {\n        const pathExp = new RegExp(limit.path);\n        if (pathExp.test(req.url)) {\n          await limit.handler(req, res, err => {\n            if (err) {\n              if (err.code === Parse.Error.CONNECTION_FAILED) {\n                throw err;\n              }\n              req.config.loggerController.error(\n                'An unknown error occured when attempting to apply the rate limiter: ',\n                err\n              );\n            }\n          });\n        }\n      })\n    );\n  } catch (error) {\n    res.status(429);\n    res.json({ code: Parse.Error.CONNECTION_FAILED, error: error.message });\n    return;\n  }\n  next();\n};\n\nexport const handleParseSession = async (req, res, next) => {\n  try {\n    const info = req.info;\n    if (req.auth) {\n      next();\n      return;\n    }\n    let requestAuth = null;\n    if (\n      info.sessionToken &&\n      req.url === '/upgradeToRevocableSession' &&\n      info.sessionToken.indexOf('r:') != 0\n    ) {\n      requestAuth = await auth.getAuthForLegacySessionToken({\n        config: req.config,\n        installationId: info.installationId,\n        sessionToken: info.sessionToken,\n      });\n    } else {\n      requestAuth = await auth.getAuthForSessionToken({\n        config: req.config,\n        installationId: info.installationId,\n        sessionToken: info.sessionToken,\n      });\n    }\n    req.auth = requestAuth;\n    next();\n  } catch (error) {\n    if (error instanceof Parse.Error) {\n      next(error);\n      return;\n    }\n    // TODO: Determine the correct error scenario.\n    req.config.loggerController.error('error getting auth for sessionToken', error);\n    throw new Parse.Error(Parse.Error.UNKNOWN_ERROR, error);\n  }\n};\n\nfunction getClientIp(req) {\n  return req.ip;\n}\n\nfunction httpAuth(req) {\n  if (!(req.req || req).headers.authorization) return;\n\n  var header = (req.req || req).headers.authorization;\n  var appId, masterKey, javascriptKey;\n\n  // parse header\n  var authPrefix = 'basic ';\n\n  var match = header.toLowerCase().indexOf(authPrefix);\n\n  if (match == 0) {\n    var encodedAuth = header.substring(authPrefix.length, header.length);\n    var credentials = decodeBase64(encodedAuth).split(':');\n\n    if (credentials.length == 2) {\n      appId = credentials[0];\n      var key = credentials[1];\n\n      var jsKeyPrefix = 'javascript-key=';\n\n      var matchKey = key.indexOf(jsKeyPrefix);\n      if (matchKey == 0) {\n        javascriptKey = key.substring(jsKeyPrefix.length, key.length);\n      } else {\n        masterKey = key;\n      }\n    }\n  }\n\n  return { appId: appId, masterKey: masterKey, javascriptKey: javascriptKey };\n}\n\nfunction decodeBase64(str) {\n  return Buffer.from(str, 'base64').toString();\n}\n\nexport function allowCrossDomain(appId) {\n  return (req, res, next) => {\n    const config = Config.get(appId, getMountForRequest(req));\n    let allowHeaders = DEFAULT_ALLOWED_HEADERS;\n    if (config && config.allowHeaders) {\n      allowHeaders += `, ${config.allowHeaders.join(', ')}`;\n    }\n    const allowOrigin = (config && config.allowOrigin) || '*';\n    res.header('Access-Control-Allow-Origin', allowOrigin);\n    res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');\n    res.header('Access-Control-Allow-Headers', allowHeaders);\n    res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id');\n    // intercept OPTIONS method\n    if ('OPTIONS' == req.method) {\n      res.sendStatus(200);\n    } else {\n      next();\n    }\n  };\n}\n\nexport function allowMethodOverride(req, res, next) {\n  if (req.method === 'POST' && req.body._method) {\n    req.originalMethod = req.method;\n    req.method = req.body._method;\n    delete req.body._method;\n  }\n  next();\n}\n\nexport function handleParseErrors(err, req, res, next) {\n  const log = (req.config && req.config.loggerController) || defaultLogger;\n  if (err instanceof Parse.Error) {\n    if (req.config && req.config.enableExpressErrorHandler) {\n      return next(err);\n    }\n    let httpStatus;\n    // TODO: fill out this mapping\n    switch (err.code) {\n      case Parse.Error.INTERNAL_SERVER_ERROR:\n        httpStatus = 500;\n        break;\n      case Parse.Error.OBJECT_NOT_FOUND:\n        httpStatus = 404;\n        break;\n      default:\n        httpStatus = 400;\n    }\n    res.status(httpStatus);\n    res.json({ code: err.code, error: err.message });\n    log.error('Parse error: ', err);\n  } else if (err.status && err.message) {\n    res.status(err.status);\n    res.json({ error: err.message });\n    if (!(process && process.env.TESTING)) {\n      next(err);\n    }\n  } else {\n    log.error('Uncaught internal server error.', err, err.stack);\n    res.status(500);\n    res.json({\n      code: Parse.Error.INTERNAL_SERVER_ERROR,\n      message: 'Internal server error.',\n    });\n    if (!(process && process.env.TESTING)) {\n      next(err);\n    }\n  }\n}\n\nexport function enforceMasterKeyAccess(req, res, next) {\n  if (!req.auth.isMaster) {\n    res.status(403);\n    res.end('{\"error\":\"unauthorized: master key is required\"}');\n    return;\n  }\n  next();\n}\n\nexport function promiseEnforceMasterKeyAccess(request) {\n  if (!request.auth.isMaster) {\n    const error = new Error();\n    error.status = 403;\n    error.message = 'unauthorized: master key is required';\n    throw error;\n  }\n  return Promise.resolve();\n}\n\nexport const addRateLimit = (route, config) => {\n  if (typeof config === 'string') {\n    config = Config.get(config);\n  }\n  for (const key in route) {\n    if (!RateLimitOptions[key]) {\n      throw `Invalid rate limit option \"${key}\"`;\n    }\n  }\n  if (!config.rateLimits) {\n    config.rateLimits = [];\n  }\n  config.rateLimits.push({\n    path: pathToRegexp(route.requestPath),\n    handler: rateLimit({\n      windowMs: route.requestTimeWindow,\n      max: route.requestCount,\n      message: route.errorResponseMessage || RateLimitOptions.errorResponseMessage.default,\n      handler: (request, response, next, options) => {\n        throw {\n          code: Parse.Error.CONNECTION_FAILED,\n          message: options.message,\n        };\n      },\n      skip: request => {\n        if (request.ip === '127.0.0.1' && !route.includeInternalRequests) {\n          return true;\n        }\n        if (route.includeMasterKey) {\n          return false;\n        }\n        if (route.requestMethods) {\n          if (Array.isArray(route.requestMethods)) {\n            if (!route.requestMethods.includes(request.method)) {\n              return true;\n            }\n          } else {\n            const regExp = new RegExp(route.requestMethods);\n            if (!regExp.test(request.method)) {\n              return true;\n            }\n          }\n        }\n        return request.auth?.isMaster;\n      },\n      keyGenerator: request => {\n        return request.config.ip;\n      },\n    }),\n  });\n  Config.put(config);\n};\n\n/**\n * Deduplicates a request to ensure idempotency. Duplicates are determined by the request ID\n * in the request header. If a request has no request ID, it is executed anyway.\n * @param {*} req The request to evaluate.\n * @returns Promise<{}>\n */\nexport function promiseEnsureIdempotency(req) {\n  // Enable feature only for MongoDB\n  if (\n    !(\n      req.config.database.adapter instanceof MongoStorageAdapter ||\n      req.config.database.adapter instanceof PostgresStorageAdapter\n    )\n  ) {\n    return Promise.resolve();\n  }\n  // Get parameters\n  const config = req.config;\n  const requestId = ((req || {}).headers || {})['x-parse-request-id'];\n  const { paths, ttl } = config.idempotencyOptions;\n  if (!requestId || !config.idempotencyOptions) {\n    return Promise.resolve();\n  }\n  // Request path may contain trailing slashes, depending on the original request, so remove\n  // leading and trailing slashes to make it easier to specify paths in the configuration\n  const reqPath = req.path.replace(/^\\/|\\/$/, '');\n  // Determine whether idempotency is enabled for current request path\n  let match = false;\n  for (const path of paths) {\n    // Assume one wants a path to always match from the beginning to prevent any mistakes\n    const regex = new RegExp(path.charAt(0) === '^' ? path : '^' + path);\n    if (reqPath.match(regex)) {\n      match = true;\n      break;\n    }\n  }\n  if (!match) {\n    return Promise.resolve();\n  }\n  // Try to store request\n  const expiryDate = new Date(new Date().setSeconds(new Date().getSeconds() + ttl));\n  return rest\n    .create(config, auth.master(config), '_Idempotency', {\n      reqId: requestId,\n      expire: Parse._encode(expiryDate),\n    })\n    .catch(e => {\n      if (e.code == Parse.Error.DUPLICATE_VALUE) {\n        throw new Parse.Error(Parse.Error.DUPLICATE_REQUEST, 'Duplicate request');\n      }\n      throw e;\n    });\n}\n\nfunction invalidRequest(req, res) {\n  res.status(403);\n  res.end('{\"error\":\"unauthorized\"}');\n}\n\nfunction malformedContext(req, res) {\n  res.status(400);\n  res.json({ code: Parse.Error.INVALID_JSON, error: 'Invalid object for context.' });\n}\n"],"mappings":"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"}