parse-server 2.8.4 → 8.6.2

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