parse-server 6.0.0-alpha.3 → 6.0.0-alpha.31

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 +58 -34
  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 +46 -87
  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 +13 -10
  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 +77 -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 -8
  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 +90 -10
  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 +178 -202
  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 +59 -156
  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 +170 -151
  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 +25 -22
  192. package/PATENTS +0 -37
@@ -3,58 +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"));
27
+ var _ipRangeCheck = _interopRequireDefault(require("ip-range-check"));
33
28
  function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
34
-
35
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';
36
30
  exports.DEFAULT_ALLOWED_HEADERS = DEFAULT_ALLOWED_HEADERS;
37
-
38
31
  const getMountForRequest = function (req) {
39
32
  const mountPathLength = req.originalUrl.length - req.url.length;
40
33
  const mountPath = req.originalUrl.slice(0, mountPathLength);
41
34
  return req.protocol + '://' + req.get('host') + mountPath;
42
- }; // 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
43
38
  // auth too.
44
39
  // The bodyparser should run before this middleware.
45
40
  // Adds info to the request:
46
41
  // req.config - the Config for this app
47
42
  // req.auth - the Auth for this request
48
-
49
-
50
43
  function handleParseHeaders(req, res, next) {
51
44
  var mount = getMountForRequest(req);
52
45
  let context = {};
53
-
54
46
  if (req.get('X-Parse-Cloud-Context') != null) {
55
47
  try {
56
48
  context = JSON.parse(req.get('X-Parse-Cloud-Context'));
57
-
58
49
  if (Object.prototype.toString.call(context) !== '[object Object]') {
59
50
  throw 'Context is not an object';
60
51
  }
@@ -62,11 +53,11 @@ function handleParseHeaders(req, res, next) {
62
53
  return malformedContext(req, res);
63
54
  }
64
55
  }
65
-
66
56
  var info = {
67
57
  appId: req.get('X-Parse-Application-Id'),
68
58
  sessionToken: req.get('X-Parse-Session-Token'),
69
59
  masterKey: req.get('X-Parse-Master-Key'),
60
+ maintenanceKey: req.get('X-Parse-Maintenance-Key'),
70
61
  installationId: req.get('X-Parse-Installation-Id'),
71
62
  clientKey: req.get('X-Parse-Client-Key'),
72
63
  javascriptKey: req.get('X-Parse-Javascript-Key'),
@@ -76,25 +67,20 @@ function handleParseHeaders(req, res, next) {
76
67
  context: context
77
68
  };
78
69
  var basicAuth = httpAuth(req);
79
-
80
70
  if (basicAuth) {
81
71
  var basicAuthAppId = basicAuth.appId;
82
-
83
72
  if (_cache.default.get(basicAuthAppId)) {
84
73
  info.appId = basicAuthAppId;
85
74
  info.masterKey = basicAuth.masterKey || info.masterKey;
86
75
  info.javascriptKey = basicAuth.javascriptKey || info.javascriptKey;
87
76
  }
88
77
  }
89
-
90
78
  if (req.body) {
91
79
  // Unity SDK sends a _noBody key which needs to be removed.
92
80
  // Unclear at this point if action needs to be taken.
93
81
  delete req.body._noBody;
94
82
  }
95
-
96
83
  var fileViaJSON = false;
97
-
98
84
  if (!info.appId || !_cache.default.get(info.appId)) {
99
85
  // See if we can find the app id on the body.
100
86
  if (req.body instanceof Buffer) {
@@ -108,48 +94,40 @@ function handleParseHeaders(req, res, next) {
108
94
  } catch (e) {
109
95
  return invalidRequest(req, res);
110
96
  }
111
-
112
97
  fileViaJSON = true;
113
98
  }
114
-
115
99
  if (req.body) {
116
100
  delete req.body._RevocableSession;
117
101
  }
118
-
119
102
  if (req.body && req.body._ApplicationId && _cache.default.get(req.body._ApplicationId) && (!info.masterKey || _cache.default.get(req.body._ApplicationId).masterKey === info.masterKey)) {
120
103
  info.appId = req.body._ApplicationId;
121
104
  info.javascriptKey = req.body._JavaScriptKey || '';
122
105
  delete req.body._ApplicationId;
123
- 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
124
108
  // SDKs are handled ok
125
-
126
109
  if (req.body._ClientVersion) {
127
110
  info.clientVersion = req.body._ClientVersion;
128
111
  delete req.body._ClientVersion;
129
112
  }
130
-
131
113
  if (req.body._InstallationId) {
132
114
  info.installationId = req.body._InstallationId;
133
115
  delete req.body._InstallationId;
134
116
  }
135
-
136
117
  if (req.body._SessionToken) {
137
118
  info.sessionToken = req.body._SessionToken;
138
119
  delete req.body._SessionToken;
139
120
  }
140
-
141
121
  if (req.body._MasterKey) {
142
122
  info.masterKey = req.body._MasterKey;
143
123
  delete req.body._MasterKey;
144
124
  }
145
-
146
125
  if (req.body._context) {
147
126
  if (req.body._context instanceof Object) {
148
127
  info.context = req.body._context;
149
128
  } else {
150
129
  try {
151
130
  info.context = JSON.parse(req.body._context);
152
-
153
131
  if (Object.prototype.toString.call(info.context) !== '[object Object]') {
154
132
  throw 'Context is not an object';
155
133
  }
@@ -157,10 +135,8 @@ function handleParseHeaders(req, res, next) {
157
135
  return malformedContext(req, res);
158
136
  }
159
137
  }
160
-
161
138
  delete req.body._context;
162
139
  }
163
-
164
140
  if (req.body._ContentType) {
165
141
  req.headers['content-type'] = req.body._ContentType;
166
142
  delete req.body._ContentType;
@@ -169,47 +145,64 @@ function handleParseHeaders(req, res, next) {
169
145
  return invalidRequest(req, res);
170
146
  }
171
147
  }
172
-
173
148
  if (info.sessionToken && typeof info.sessionToken !== 'string') {
174
149
  info.sessionToken = info.sessionToken.toString();
175
150
  }
176
-
177
151
  if (info.clientVersion) {
178
152
  info.clientSDK = _ClientSDK.default.fromString(info.clientVersion);
179
153
  }
180
-
181
154
  if (fileViaJSON) {
182
- req.fileData = req.body.fileData; // We need to repopulate req.body with a buffer
183
-
155
+ req.fileData = req.body.fileData;
156
+ // We need to repopulate req.body with a buffer
184
157
  var base64 = req.body.base64;
185
158
  req.body = Buffer.from(base64, 'base64');
186
159
  }
187
-
188
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
+ }
189
170
  info.app = _cache.default.get(info.appId);
190
- req.config = _Config.default.get(info.appId, mount);
171
+ req.config = config;
191
172
  req.config.headers = req.headers || {};
192
173
  req.config.ip = clientIp;
193
174
  req.info = info;
194
-
195
- if (info.masterKey && req.config.masterKeyIps && req.config.masterKeyIps.length !== 0 && req.config.masterKeyIps.indexOf(clientIp) === -1) {
196
- return invalidRequest(req, res);
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
+ }
190
+ let isMaster = info.masterKey === req.config.masterKey;
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'.`);
195
+ isMaster = false;
197
196
  }
198
-
199
- var isMaster = info.masterKey === req.config.masterKey;
200
-
201
197
  if (isMaster) {
202
198
  req.auth = new _Auth.default.Auth({
203
199
  config: req.config,
204
200
  installationId: info.installationId,
205
201
  isMaster: true
206
202
  });
207
- next();
208
- return;
203
+ return handleRateLimit(req, res, next);
209
204
  }
210
-
211
205
  var isReadOnlyMaster = info.masterKey === req.config.readOnlyMasterKey;
212
-
213
206
  if (typeof req.config.readOnlyMasterKey != 'undefined' && req.config.readOnlyMasterKey && isReadOnlyMaster) {
214
207
  req.auth = new _Auth.default.Auth({
215
208
  config: req.config,
@@ -217,12 +210,11 @@ function handleParseHeaders(req, res, next) {
217
210
  isMaster: true,
218
211
  isReadOnly: true
219
212
  });
220
- next();
221
- return;
222
- } // Client keys are not required in parse-server, but if any have been configured in the server, validate them
223
- // to preserve original behavior.
224
-
213
+ return handleRateLimit(req, res, next);
214
+ }
225
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.
226
218
  const keys = ['clientKey', 'javascriptKey', 'dotNetKey', 'restAPIKey'];
227
219
  const oneKeyConfigured = keys.some(function (key) {
228
220
  return req.config[key] !== undefined;
@@ -230,15 +222,12 @@ function handleParseHeaders(req, res, next) {
230
222
  const oneKeyMatches = keys.some(function (key) {
231
223
  return req.config[key] !== undefined && info[key] === req.config[key];
232
224
  });
233
-
234
225
  if (oneKeyConfigured && !oneKeyMatches) {
235
226
  return invalidRequest(req, res);
236
227
  }
237
-
238
228
  if (req.url == '/login') {
239
229
  delete info.sessionToken;
240
230
  }
241
-
242
231
  if (req.userFromJWT) {
243
232
  req.auth = new _Auth.default.Auth({
244
233
  config: req.config,
@@ -246,89 +235,96 @@ function handleParseHeaders(req, res, next) {
246
235
  isMaster: false,
247
236
  user: req.userFromJWT
248
237
  });
249
- next();
250
- return;
238
+ return handleRateLimit(req, res, next);
251
239
  }
252
-
253
240
  if (!info.sessionToken) {
254
241
  req.auth = new _Auth.default.Auth({
255
242
  config: req.config,
256
243
  installationId: info.installationId,
257
244
  isMaster: false
258
245
  });
259
- 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
+ });
260
271
  return;
261
272
  }
262
-
263
- return Promise.resolve().then(() => {
264
- // 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;
265
283
  if (info.sessionToken && req.url === '/upgradeToRevocableSession' && info.sessionToken.indexOf('r:') != 0) {
266
- return _Auth.default.getAuthForLegacySessionToken({
284
+ requestAuth = await _Auth.default.getAuthForLegacySessionToken({
267
285
  config: req.config,
268
286
  installationId: info.installationId,
269
287
  sessionToken: info.sessionToken
270
288
  });
271
289
  } else {
272
- return _Auth.default.getAuthForSessionToken({
290
+ requestAuth = await _Auth.default.getAuthForSessionToken({
273
291
  config: req.config,
274
292
  installationId: info.installationId,
275
293
  sessionToken: info.sessionToken
276
294
  });
277
295
  }
278
- }).then(auth => {
279
- if (auth) {
280
- req.auth = auth;
281
- next();
282
- }
283
- }).catch(error => {
296
+ req.auth = requestAuth;
297
+ next();
298
+ } catch (error) {
284
299
  if (error instanceof _node.default.Error) {
285
300
  next(error);
286
301
  return;
287
- } else {
288
- // TODO: Determine the correct error scenario.
289
- req.config.loggerController.error('error getting auth for sessionToken', error);
290
- throw new _node.default.Error(_node.default.Error.UNKNOWN_ERROR, error);
291
302
  }
292
- });
293
- }
294
-
295
- function getClientIp(req) {
296
- if (req.headers['x-forwarded-for']) {
297
- // try to get from x-forwared-for if it set (behind reverse proxy)
298
- return req.headers['x-forwarded-for'].split(',')[0];
299
- } else if (req.connection && req.connection.remoteAddress) {
300
- // no proxy, try getting from connection.remoteAddress
301
- return req.connection.remoteAddress;
302
- } else if (req.socket) {
303
- // try to get it from req.socket
304
- return req.socket.remoteAddress;
305
- } else if (req.connection && req.connection.socket) {
306
- // try to get it form the connection.socket
307
- return req.connection.socket.remoteAddress;
308
- } else {
309
- // if non above, fallback.
310
- 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);
311
306
  }
307
+ };
308
+ exports.handleParseSession = handleParseSession;
309
+ function getClientIp(req) {
310
+ return req.ip;
312
311
  }
313
-
314
312
  function httpAuth(req) {
315
313
  if (!(req.req || req).headers.authorization) return;
316
314
  var header = (req.req || req).headers.authorization;
317
- var appId, masterKey, javascriptKey; // parse header
315
+ var appId, masterKey, javascriptKey;
318
316
 
317
+ // parse header
319
318
  var authPrefix = 'basic ';
320
319
  var match = header.toLowerCase().indexOf(authPrefix);
321
-
322
320
  if (match == 0) {
323
321
  var encodedAuth = header.substring(authPrefix.length, header.length);
324
322
  var credentials = decodeBase64(encodedAuth).split(':');
325
-
326
323
  if (credentials.length == 2) {
327
324
  appId = credentials[0];
328
325
  var key = credentials[1];
329
326
  var jsKeyPrefix = 'javascript-key=';
330
327
  var matchKey = key.indexOf(jsKeyPrefix);
331
-
332
328
  if (matchKey == 0) {
333
329
  javascriptKey = key.substring(jsKeyPrefix.length, key.length);
334
330
  } else {
@@ -336,34 +332,28 @@ function httpAuth(req) {
336
332
  }
337
333
  }
338
334
  }
339
-
340
335
  return {
341
336
  appId: appId,
342
337
  masterKey: masterKey,
343
338
  javascriptKey: javascriptKey
344
339
  };
345
340
  }
346
-
347
341
  function decodeBase64(str) {
348
342
  return Buffer.from(str, 'base64').toString();
349
343
  }
350
-
351
344
  function allowCrossDomain(appId) {
352
345
  return (req, res, next) => {
353
346
  const config = _Config.default.get(appId, getMountForRequest(req));
354
-
355
347
  let allowHeaders = DEFAULT_ALLOWED_HEADERS;
356
-
357
348
  if (config && config.allowHeaders) {
358
349
  allowHeaders += `, ${config.allowHeaders.join(', ')}`;
359
350
  }
360
-
361
351
  const allowOrigin = config && config.allowOrigin || '*';
362
352
  res.header('Access-Control-Allow-Origin', allowOrigin);
363
353
  res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');
364
354
  res.header('Access-Control-Allow-Headers', allowHeaders);
365
- res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id'); // intercept OPTIONS method
366
-
355
+ res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id');
356
+ // intercept OPTIONS method
367
357
  if ('OPTIONS' == req.method) {
368
358
  res.sendStatus(200);
369
359
  } else {
@@ -371,40 +361,32 @@ function allowCrossDomain(appId) {
371
361
  }
372
362
  };
373
363
  }
374
-
375
364
  function allowMethodOverride(req, res, next) {
376
365
  if (req.method === 'POST' && req.body._method) {
377
366
  req.originalMethod = req.method;
378
367
  req.method = req.body._method;
379
368
  delete req.body._method;
380
369
  }
381
-
382
370
  next();
383
371
  }
384
-
385
372
  function handleParseErrors(err, req, res, next) {
386
373
  const log = req.config && req.config.loggerController || _logger.default;
387
-
388
374
  if (err instanceof _node.default.Error) {
389
375
  if (req.config && req.config.enableExpressErrorHandler) {
390
376
  return next(err);
391
377
  }
392
-
393
- let httpStatus; // TODO: fill out this mapping
394
-
378
+ let httpStatus;
379
+ // TODO: fill out this mapping
395
380
  switch (err.code) {
396
381
  case _node.default.Error.INTERNAL_SERVER_ERROR:
397
382
  httpStatus = 500;
398
383
  break;
399
-
400
384
  case _node.default.Error.OBJECT_NOT_FOUND:
401
385
  httpStatus = 404;
402
386
  break;
403
-
404
387
  default:
405
388
  httpStatus = 400;
406
389
  }
407
-
408
390
  res.status(httpStatus);
409
391
  res.json({
410
392
  code: err.code,
@@ -416,7 +398,6 @@ function handleParseErrors(err, req, res, next) {
416
398
  res.json({
417
399
  error: err.message
418
400
  });
419
-
420
401
  if (!(process && process.env.TESTING)) {
421
402
  next(err);
422
403
  }
@@ -427,23 +408,19 @@ function handleParseErrors(err, req, res, next) {
427
408
  code: _node.default.Error.INTERNAL_SERVER_ERROR,
428
409
  message: 'Internal server error.'
429
410
  });
430
-
431
411
  if (!(process && process.env.TESTING)) {
432
412
  next(err);
433
413
  }
434
414
  }
435
415
  }
436
-
437
416
  function enforceMasterKeyAccess(req, res, next) {
438
417
  if (!req.auth.isMaster) {
439
418
  res.status(403);
440
419
  res.end('{"error":"unauthorized: master key is required"}');
441
420
  return;
442
421
  }
443
-
444
422
  next();
445
423
  }
446
-
447
424
  function promiseEnforceMasterKeyAccess(request) {
448
425
  if (!request.auth.isMaster) {
449
426
  const error = new Error();
@@ -451,56 +428,101 @@ function promiseEnforceMasterKeyAccess(request) {
451
428
  error.message = 'unauthorized: master key is required';
452
429
  throw error;
453
430
  }
454
-
455
431
  return Promise.resolve();
456
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
+
457
487
  /**
458
488
  * Deduplicates a request to ensure idempotency. Duplicates are determined by the request ID
459
489
  * in the request header. If a request has no request ID, it is executed anyway.
460
490
  * @param {*} req The request to evaluate.
461
491
  * @returns Promise<{}>
462
492
  */
463
-
464
-
493
+ exports.addRateLimit = addRateLimit;
465
494
  function promiseEnsureIdempotency(req) {
466
495
  // Enable feature only for MongoDB
467
496
  if (!(req.config.database.adapter instanceof _MongoStorageAdapter.default || req.config.database.adapter instanceof _PostgresStorageAdapter.default)) {
468
497
  return Promise.resolve();
469
- } // Get parameters
470
-
471
-
498
+ }
499
+ // Get parameters
472
500
  const config = req.config;
473
501
  const requestId = ((req || {}).headers || {})['x-parse-request-id'];
474
502
  const {
475
503
  paths,
476
504
  ttl
477
505
  } = config.idempotencyOptions;
478
-
479
506
  if (!requestId || !config.idempotencyOptions) {
480
507
  return Promise.resolve();
481
- } // 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
482
510
  // leading and trailing slashes to make it easier to specify paths in the configuration
483
-
484
-
485
- const reqPath = req.path.replace(/^\/|\/$/, ''); // Determine whether idempotency is enabled for current request path
486
-
511
+ const reqPath = req.path.replace(/^\/|\/$/, '');
512
+ // Determine whether idempotency is enabled for current request path
487
513
  let match = false;
488
-
489
514
  for (const path of paths) {
490
515
  // Assume one wants a path to always match from the beginning to prevent any mistakes
491
516
  const regex = new RegExp(path.charAt(0) === '^' ? path : '^' + path);
492
-
493
517
  if (reqPath.match(regex)) {
494
518
  match = true;
495
519
  break;
496
520
  }
497
521
  }
498
-
499
522
  if (!match) {
500
523
  return Promise.resolve();
501
- } // Try to store request
502
-
503
-
524
+ }
525
+ // Try to store request
504
526
  const expiryDate = new Date(new Date().setSeconds(new Date().getSeconds() + ttl));
505
527
  return _rest.default.create(config, _Auth.default.master(config), '_Idempotency', {
506
528
  reqId: requestId,
@@ -509,16 +531,13 @@ function promiseEnsureIdempotency(req) {
509
531
  if (e.code == _node.default.Error.DUPLICATE_VALUE) {
510
532
  throw new _node.default.Error(_node.default.Error.DUPLICATE_REQUEST, 'Duplicate request');
511
533
  }
512
-
513
534
  throw e;
514
535
  });
515
536
  }
516
-
517
537
  function invalidRequest(req, res) {
518
538
  res.status(403);
519
539
  res.end('{"error":"unauthorized"}');
520
540
  }
521
-
522
541
  function malformedContext(req, res) {
523
542
  res.status(400);
524
543
  res.json({
@@ -526,4 +545,4 @@ function malformedContext(req, res) {
526
545
  error: 'Invalid object for context.'
527
546
  });
528
547
  }
529
- //# 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","masterKeyIps","indexOf","isMaster","auth","Auth","isReadOnlyMaster","readOnlyMasterKey","isReadOnly","keys","oneKeyConfigured","some","key","undefined","oneKeyMatches","userFromJWT","user","Promise","resolve","then","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';\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  if (\n    info.masterKey &&\n    req.config.masterKeyIps &&\n    req.config.masterKeyIps.length !== 0 &&\n    req.config.masterKeyIps.indexOf(clientIp) === -1\n  ) {\n    return invalidRequest(req, res);\n  }\n\n  var isMaster = info.masterKey === req.config.masterKey;\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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"}