parse-server 6.0.0-alpha.2 → 6.0.0-alpha.21

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 (191) hide show
  1. package/README.md +45 -17
  2. package/lib/AccountLockout.js +11 -26
  3. package/lib/Adapters/AdapterLoader.js +8 -14
  4. package/lib/Adapters/Analytics/AnalyticsAdapter.js +2 -8
  5. package/lib/Adapters/Auth/AuthAdapter.js +7 -16
  6. package/lib/Adapters/Auth/OAuth1Client.js +32 -57
  7. package/lib/Adapters/Auth/apple.js +6 -22
  8. package/lib/Adapters/Auth/facebook.js +7 -37
  9. package/lib/Adapters/Auth/gcenter.js +8 -37
  10. package/lib/Adapters/Auth/github.js +7 -10
  11. package/lib/Adapters/Auth/google.js +11 -34
  12. package/lib/Adapters/Auth/gpgames.js +5 -8
  13. package/lib/Adapters/Auth/httpsRequest.js +1 -7
  14. package/lib/Adapters/Auth/index.js +20 -65
  15. package/lib/Adapters/Auth/instagram.js +5 -9
  16. package/lib/Adapters/Auth/janraincapture.js +8 -12
  17. package/lib/Adapters/Auth/janrainengage.js +7 -11
  18. package/lib/Adapters/Auth/keycloak.js +5 -19
  19. package/lib/Adapters/Auth/ldap.js +1 -15
  20. package/lib/Adapters/Auth/line.js +7 -10
  21. package/lib/Adapters/Auth/linkedin.js +7 -12
  22. package/lib/Adapters/Auth/meetup.js +7 -10
  23. package/lib/Adapters/Auth/microsoft.js +7 -10
  24. package/lib/Adapters/Auth/oauth2.js +6 -18
  25. package/lib/Adapters/Auth/phantauth.js +8 -10
  26. package/lib/Adapters/Auth/qq.js +7 -13
  27. package/lib/Adapters/Auth/spotify.js +7 -14
  28. package/lib/Adapters/Auth/twitter.js +5 -15
  29. package/lib/Adapters/Auth/vkontakte.js +9 -15
  30. package/lib/Adapters/Auth/wechat.js +7 -10
  31. package/lib/Adapters/Auth/weibo.js +7 -11
  32. package/lib/Adapters/Cache/CacheAdapter.js +4 -12
  33. package/lib/Adapters/Cache/InMemoryCache.js +5 -19
  34. package/lib/Adapters/Cache/InMemoryCacheAdapter.js +1 -11
  35. package/lib/Adapters/Cache/LRUCache.js +1 -11
  36. package/lib/Adapters/Cache/NullCacheAdapter.js +1 -8
  37. package/lib/Adapters/Cache/RedisCacheAdapter.js +46 -87
  38. package/lib/Adapters/Cache/SchemaCache.js +1 -6
  39. package/lib/Adapters/Email/MailAdapter.js +2 -7
  40. package/lib/Adapters/Files/FilesAdapter.js +7 -21
  41. package/lib/Adapters/Files/GridFSBucketAdapter.js +6 -44
  42. package/lib/Adapters/Files/GridStoreAdapter.js +1 -1
  43. package/lib/Adapters/Logger/LoggerAdapter.js +2 -11
  44. package/lib/Adapters/Logger/WinstonLogger.js +3 -30
  45. package/lib/Adapters/Logger/WinstonLoggerAdapter.js +5 -16
  46. package/lib/Adapters/MessageQueue/EventEmitterMQ.js +3 -20
  47. package/lib/Adapters/PubSub/EventEmitterPubSub.js +1 -16
  48. package/lib/Adapters/PubSub/PubSubAdapter.js +2 -9
  49. package/lib/Adapters/PubSub/RedisPubSub.js +13 -10
  50. package/lib/Adapters/Push/PushAdapter.js +2 -8
  51. package/lib/Adapters/Storage/Mongo/MongoCollection.js +12 -37
  52. package/lib/Adapters/Storage/Mongo/MongoSchemaCollection.js +26 -79
  53. package/lib/Adapters/Storage/Mongo/MongoStorageAdapter.js +78 -209
  54. package/lib/Adapters/Storage/Mongo/MongoTransform.js +82 -371
  55. package/lib/Adapters/Storage/Postgres/PostgresClient.js +1 -13
  56. package/lib/Adapters/Storage/Postgres/PostgresConfigParser.js +1 -20
  57. package/lib/Adapters/Storage/Postgres/PostgresStorageAdapter.js +119 -446
  58. package/lib/Adapters/Storage/Postgres/sql/index.js +4 -7
  59. package/lib/Adapters/Storage/StorageAdapter.js +1 -1
  60. package/lib/Adapters/WebSocketServer/WSAdapter.js +3 -12
  61. package/lib/Adapters/WebSocketServer/WSSAdapter.js +7 -12
  62. package/lib/Auth.js +54 -121
  63. package/lib/ClientSDK.js +3 -11
  64. package/lib/Config.js +69 -113
  65. package/lib/Controllers/AdaptableController.js +6 -18
  66. package/lib/Controllers/AnalyticsController.js +1 -9
  67. package/lib/Controllers/CacheController.js +3 -23
  68. package/lib/Controllers/DatabaseController.js +147 -345
  69. package/lib/Controllers/FilesController.js +5 -34
  70. package/lib/Controllers/HooksController.js +1 -51
  71. package/lib/Controllers/LiveQueryController.js +4 -23
  72. package/lib/Controllers/LoggerController.js +15 -54
  73. package/lib/Controllers/ParseGraphQLController.js +49 -104
  74. package/lib/Controllers/PushController.js +20 -59
  75. package/lib/Controllers/SchemaController.js +154 -344
  76. package/lib/Controllers/UserController.js +11 -72
  77. package/lib/Controllers/index.js +19 -68
  78. package/lib/Controllers/types.js +1 -1
  79. package/lib/Deprecator/Deprecations.js +1 -8
  80. package/lib/Deprecator/Deprecator.js +9 -18
  81. package/lib/GraphQL/ParseGraphQLSchema.js +16 -100
  82. package/lib/GraphQL/ParseGraphQLServer.js +2 -29
  83. package/lib/GraphQL/helpers/objectsMutations.js +2 -12
  84. package/lib/GraphQL/helpers/objectsQueries.js +18 -76
  85. package/lib/GraphQL/loaders/defaultGraphQLMutations.js +1 -9
  86. package/lib/GraphQL/loaders/defaultGraphQLQueries.js +1 -8
  87. package/lib/GraphQL/loaders/defaultGraphQLTypes.js +9 -115
  88. package/lib/GraphQL/loaders/defaultRelaySchema.js +6 -18
  89. package/lib/GraphQL/loaders/filesMutations.js +2 -19
  90. package/lib/GraphQL/loaders/functionsMutations.js +6 -17
  91. package/lib/GraphQL/loaders/parseClassMutations.js +6 -44
  92. package/lib/GraphQL/loaders/parseClassQueries.js +1 -26
  93. package/lib/GraphQL/loaders/parseClassTypes.js +10 -64
  94. package/lib/GraphQL/loaders/schemaDirectives.js +1 -17
  95. package/lib/GraphQL/loaders/schemaMutations.js +1 -20
  96. package/lib/GraphQL/loaders/schemaQueries.js +1 -14
  97. package/lib/GraphQL/loaders/schemaTypes.js +2 -6
  98. package/lib/GraphQL/loaders/usersMutations.js +6 -28
  99. package/lib/GraphQL/loaders/usersQueries.js +4 -26
  100. package/lib/GraphQL/parseGraphQLUtils.js +6 -19
  101. package/lib/GraphQL/transformers/className.js +1 -4
  102. package/lib/GraphQL/transformers/constraintType.js +1 -20
  103. package/lib/GraphQL/transformers/inputType.js +1 -20
  104. package/lib/GraphQL/transformers/mutation.js +6 -51
  105. package/lib/GraphQL/transformers/outputType.js +1 -20
  106. package/lib/GraphQL/transformers/query.js +6 -42
  107. package/lib/GraphQL/transformers/schemaFields.js +7 -34
  108. package/lib/KeyPromiseQueue.js +1 -12
  109. package/lib/LiveQuery/Client.js +1 -25
  110. package/lib/LiveQuery/Id.js +1 -7
  111. package/lib/LiveQuery/ParseCloudCodePublisher.js +13 -19
  112. package/lib/LiveQuery/ParseLiveQueryServer.js +92 -306
  113. package/lib/LiveQuery/ParsePubSub.js +1 -12
  114. package/lib/LiveQuery/ParseWebSocketServer.js +4 -26
  115. package/lib/LiveQuery/QueryTools.js +14 -116
  116. package/lib/LiveQuery/RequestSchema.js +1 -1
  117. package/lib/LiveQuery/SessionTokenCache.js +1 -17
  118. package/lib/LiveQuery/Subscription.js +4 -18
  119. package/lib/LiveQuery/equalObjects.js +2 -14
  120. package/lib/Options/Definitions.js +79 -10
  121. package/lib/Options/docs.js +23 -3
  122. package/lib/Options/index.js +4 -12
  123. package/lib/Options/parsers.js +1 -18
  124. package/lib/Page.js +1 -9
  125. package/lib/ParseMessageQueue.js +1 -10
  126. package/lib/ParseServer.js +144 -182
  127. package/lib/ParseServerRESTController.js +6 -33
  128. package/lib/PromiseRouter.js +16 -50
  129. package/lib/Push/PushQueue.js +3 -15
  130. package/lib/Push/PushWorker.js +7 -32
  131. package/lib/Push/utils.js +9 -38
  132. package/lib/RestQuery.js +105 -242
  133. package/lib/RestWrite.js +212 -377
  134. package/lib/Routers/AggregateRouter.js +14 -51
  135. package/lib/Routers/AnalyticsRouter.js +2 -8
  136. package/lib/Routers/AudiencesRouter.js +1 -15
  137. package/lib/Routers/ClassesRouter.js +3 -53
  138. package/lib/Routers/CloudCodeRouter.js +1 -19
  139. package/lib/Routers/FeaturesRouter.js +1 -10
  140. package/lib/Routers/FilesRouter.js +29 -76
  141. package/lib/Routers/FunctionsRouter.js +5 -28
  142. package/lib/Routers/GlobalConfigRouter.js +4 -18
  143. package/lib/Routers/GraphQLRouter.js +1 -14
  144. package/lib/Routers/HooksRouter.js +1 -29
  145. package/lib/Routers/IAPValidationRouter.js +6 -29
  146. package/lib/Routers/InstallationsRouter.js +2 -12
  147. package/lib/Routers/LogsRouter.js +4 -16
  148. package/lib/Routers/PagesRouter.js +69 -129
  149. package/lib/Routers/PublicAPIRouter.js +3 -62
  150. package/lib/Routers/PurgeRouter.js +1 -15
  151. package/lib/Routers/PushRouter.js +2 -18
  152. package/lib/Routers/RolesRouter.js +1 -7
  153. package/lib/Routers/SchemasRouter.js +4 -34
  154. package/lib/Routers/SecurityRouter.js +1 -12
  155. package/lib/Routers/SessionsRouter.js +3 -19
  156. package/lib/Routers/UsersRouter.js +58 -155
  157. package/lib/SchemaMigrations/DefinedSchemas.js +56 -115
  158. package/lib/SchemaMigrations/Migrations.js +2 -8
  159. package/lib/Security/Check.js +8 -16
  160. package/lib/Security/CheckGroup.js +4 -11
  161. package/lib/Security/CheckGroups/CheckGroupDatabase.js +8 -18
  162. package/lib/Security/CheckGroups/CheckGroupServerConfig.js +5 -15
  163. package/lib/Security/CheckGroups/CheckGroups.js +1 -4
  164. package/lib/Security/CheckRunner.js +22 -41
  165. package/lib/StatusHandler.js +12 -69
  166. package/lib/TestUtils.js +1 -6
  167. package/lib/Utils.js +27 -66
  168. package/lib/batch.js +17 -28
  169. package/lib/cache.js +1 -3
  170. package/lib/cli/definitions/parse-live-query-server.js +1 -3
  171. package/lib/cli/definitions/parse-server.js +1 -3
  172. package/lib/cli/parse-live-query-server.js +1 -6
  173. package/lib/cli/parse-server.js +11 -21
  174. package/lib/cli/utils/commander.js +13 -51
  175. package/lib/cli/utils/runner.js +1 -14
  176. package/lib/cloud-code/Parse.Cloud.js +71 -92
  177. package/lib/cryptoUtils.js +11 -19
  178. package/lib/defaults.js +2 -14
  179. package/lib/deprecated.js +1 -2
  180. package/lib/index.js +16 -34
  181. package/lib/logger.js +6 -13
  182. package/lib/middlewares.js +147 -151
  183. package/lib/password.js +6 -10
  184. package/lib/request.js +173 -2
  185. package/lib/requiredParameter.js +1 -3
  186. package/lib/rest.js +19 -41
  187. package/lib/triggers.js +54 -252
  188. package/lib/vendor/mongodbUrl.js +125 -305
  189. package/package.json +22 -19
  190. package/lib/cloud-code/HTTPResponse.js +0 -73
  191. package/lib/cloud-code/httpRequest.js +0 -192
@@ -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,7 +53,6 @@ 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'),
@@ -76,25 +66,20 @@ function handleParseHeaders(req, res, next) {
76
66
  context: context
77
67
  };
78
68
  var basicAuth = httpAuth(req);
79
-
80
69
  if (basicAuth) {
81
70
  var basicAuthAppId = basicAuth.appId;
82
-
83
71
  if (_cache.default.get(basicAuthAppId)) {
84
72
  info.appId = basicAuthAppId;
85
73
  info.masterKey = basicAuth.masterKey || info.masterKey;
86
74
  info.javascriptKey = basicAuth.javascriptKey || info.javascriptKey;
87
75
  }
88
76
  }
89
-
90
77
  if (req.body) {
91
78
  // Unity SDK sends a _noBody key which needs to be removed.
92
79
  // Unclear at this point if action needs to be taken.
93
80
  delete req.body._noBody;
94
81
  }
95
-
96
82
  var fileViaJSON = false;
97
-
98
83
  if (!info.appId || !_cache.default.get(info.appId)) {
99
84
  // See if we can find the app id on the body.
100
85
  if (req.body instanceof Buffer) {
@@ -108,48 +93,40 @@ function handleParseHeaders(req, res, next) {
108
93
  } catch (e) {
109
94
  return invalidRequest(req, res);
110
95
  }
111
-
112
96
  fileViaJSON = true;
113
97
  }
114
-
115
98
  if (req.body) {
116
99
  delete req.body._RevocableSession;
117
100
  }
118
-
119
101
  if (req.body && req.body._ApplicationId && _cache.default.get(req.body._ApplicationId) && (!info.masterKey || _cache.default.get(req.body._ApplicationId).masterKey === info.masterKey)) {
120
102
  info.appId = req.body._ApplicationId;
121
103
  info.javascriptKey = req.body._JavaScriptKey || '';
122
104
  delete req.body._ApplicationId;
123
- delete req.body._JavaScriptKey; // TODO: test that the REST API formats generated by the other
105
+ delete req.body._JavaScriptKey;
106
+ // TODO: test that the REST API formats generated by the other
124
107
  // SDKs are handled ok
125
-
126
108
  if (req.body._ClientVersion) {
127
109
  info.clientVersion = req.body._ClientVersion;
128
110
  delete req.body._ClientVersion;
129
111
  }
130
-
131
112
  if (req.body._InstallationId) {
132
113
  info.installationId = req.body._InstallationId;
133
114
  delete req.body._InstallationId;
134
115
  }
135
-
136
116
  if (req.body._SessionToken) {
137
117
  info.sessionToken = req.body._SessionToken;
138
118
  delete req.body._SessionToken;
139
119
  }
140
-
141
120
  if (req.body._MasterKey) {
142
121
  info.masterKey = req.body._MasterKey;
143
122
  delete req.body._MasterKey;
144
123
  }
145
-
146
124
  if (req.body._context) {
147
125
  if (req.body._context instanceof Object) {
148
126
  info.context = req.body._context;
149
127
  } else {
150
128
  try {
151
129
  info.context = JSON.parse(req.body._context);
152
-
153
130
  if (Object.prototype.toString.call(info.context) !== '[object Object]') {
154
131
  throw 'Context is not an object';
155
132
  }
@@ -157,10 +134,8 @@ function handleParseHeaders(req, res, next) {
157
134
  return malformedContext(req, res);
158
135
  }
159
136
  }
160
-
161
137
  delete req.body._context;
162
138
  }
163
-
164
139
  if (req.body._ContentType) {
165
140
  req.headers['content-type'] = req.body._ContentType;
166
141
  delete req.body._ContentType;
@@ -169,47 +144,49 @@ function handleParseHeaders(req, res, next) {
169
144
  return invalidRequest(req, res);
170
145
  }
171
146
  }
172
-
173
147
  if (info.sessionToken && typeof info.sessionToken !== 'string') {
174
148
  info.sessionToken = info.sessionToken.toString();
175
149
  }
176
-
177
150
  if (info.clientVersion) {
178
151
  info.clientSDK = _ClientSDK.default.fromString(info.clientVersion);
179
152
  }
180
-
181
153
  if (fileViaJSON) {
182
- req.fileData = req.body.fileData; // We need to repopulate req.body with a buffer
183
-
154
+ req.fileData = req.body.fileData;
155
+ // We need to repopulate req.body with a buffer
184
156
  var base64 = req.body.base64;
185
157
  req.body = Buffer.from(base64, 'base64');
186
158
  }
187
-
188
159
  const clientIp = getClientIp(req);
160
+ const config = _Config.default.get(info.appId, mount);
161
+ if (config.state && config.state !== 'ok') {
162
+ res.status(500);
163
+ res.json({
164
+ code: _node.default.Error.INTERNAL_SERVER_ERROR,
165
+ error: `Invalid server state: ${config.state}`
166
+ });
167
+ return;
168
+ }
189
169
  info.app = _cache.default.get(info.appId);
190
- req.config = _Config.default.get(info.appId, mount);
170
+ req.config = config;
191
171
  req.config.headers = req.headers || {};
192
172
  req.config.ip = clientIp;
193
173
  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);
174
+ let isMaster = info.masterKey === req.config.masterKey;
175
+ if (isMaster && !(0, _ipRangeCheck.default)(clientIp, req.config.masterKeyIps || [])) {
176
+ var _req$config;
177
+ const log = ((_req$config = req.config) === null || _req$config === void 0 ? void 0 : _req$config.loggerController) || _logger.default;
178
+ log.error(`Request using master key rejected as the request IP address '${clientIp}' is not set in Parse Server option 'masterKeyIps'.`);
179
+ isMaster = false;
197
180
  }
198
-
199
- var isMaster = info.masterKey === req.config.masterKey;
200
-
201
181
  if (isMaster) {
202
182
  req.auth = new _Auth.default.Auth({
203
183
  config: req.config,
204
184
  installationId: info.installationId,
205
185
  isMaster: true
206
186
  });
207
- next();
208
- return;
187
+ return handleRateLimit(req, res, next);
209
188
  }
210
-
211
189
  var isReadOnlyMaster = info.masterKey === req.config.readOnlyMasterKey;
212
-
213
190
  if (typeof req.config.readOnlyMasterKey != 'undefined' && req.config.readOnlyMasterKey && isReadOnlyMaster) {
214
191
  req.auth = new _Auth.default.Auth({
215
192
  config: req.config,
@@ -217,12 +194,11 @@ function handleParseHeaders(req, res, next) {
217
194
  isMaster: true,
218
195
  isReadOnly: true
219
196
  });
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
-
197
+ return handleRateLimit(req, res, next);
198
+ }
225
199
 
200
+ // Client keys are not required in parse-server, but if any have been configured in the server, validate them
201
+ // to preserve original behavior.
226
202
  const keys = ['clientKey', 'javascriptKey', 'dotNetKey', 'restAPIKey'];
227
203
  const oneKeyConfigured = keys.some(function (key) {
228
204
  return req.config[key] !== undefined;
@@ -230,15 +206,12 @@ function handleParseHeaders(req, res, next) {
230
206
  const oneKeyMatches = keys.some(function (key) {
231
207
  return req.config[key] !== undefined && info[key] === req.config[key];
232
208
  });
233
-
234
209
  if (oneKeyConfigured && !oneKeyMatches) {
235
210
  return invalidRequest(req, res);
236
211
  }
237
-
238
212
  if (req.url == '/login') {
239
213
  delete info.sessionToken;
240
214
  }
241
-
242
215
  if (req.userFromJWT) {
243
216
  req.auth = new _Auth.default.Auth({
244
217
  config: req.config,
@@ -246,89 +219,93 @@ function handleParseHeaders(req, res, next) {
246
219
  isMaster: false,
247
220
  user: req.userFromJWT
248
221
  });
249
- next();
250
- return;
222
+ return handleRateLimit(req, res, next);
251
223
  }
252
-
253
224
  if (!info.sessionToken) {
254
225
  req.auth = new _Auth.default.Auth({
255
226
  config: req.config,
256
227
  installationId: info.installationId,
257
228
  isMaster: false
258
229
  });
259
- next();
230
+ }
231
+ handleRateLimit(req, res, next);
232
+ }
233
+ const handleRateLimit = async (req, res, next) => {
234
+ const rateLimits = req.config.rateLimits || [];
235
+ try {
236
+ await Promise.all(rateLimits.map(async limit => {
237
+ const pathExp = new RegExp(limit.path);
238
+ if (pathExp.test(req.url)) {
239
+ await limit.handler(req, res, err => {
240
+ if (err) {
241
+ throw err;
242
+ }
243
+ });
244
+ }
245
+ }));
246
+ } catch (error) {
247
+ res.status(429);
248
+ res.json({
249
+ code: _node.default.Error.CONNECTION_FAILED,
250
+ error
251
+ });
260
252
  return;
261
253
  }
262
-
263
- return Promise.resolve().then(() => {
264
- // handle the upgradeToRevocableSession path on it's own
254
+ next();
255
+ };
256
+ const handleParseSession = async (req, res, next) => {
257
+ try {
258
+ const info = req.info;
259
+ if (req.auth) {
260
+ next();
261
+ return;
262
+ }
263
+ let requestAuth = null;
265
264
  if (info.sessionToken && req.url === '/upgradeToRevocableSession' && info.sessionToken.indexOf('r:') != 0) {
266
- return _Auth.default.getAuthForLegacySessionToken({
265
+ requestAuth = await _Auth.default.getAuthForLegacySessionToken({
267
266
  config: req.config,
268
267
  installationId: info.installationId,
269
268
  sessionToken: info.sessionToken
270
269
  });
271
270
  } else {
272
- return _Auth.default.getAuthForSessionToken({
271
+ requestAuth = await _Auth.default.getAuthForSessionToken({
273
272
  config: req.config,
274
273
  installationId: info.installationId,
275
274
  sessionToken: info.sessionToken
276
275
  });
277
276
  }
278
- }).then(auth => {
279
- if (auth) {
280
- req.auth = auth;
281
- next();
282
- }
283
- }).catch(error => {
277
+ req.auth = requestAuth;
278
+ next();
279
+ } catch (error) {
284
280
  if (error instanceof _node.default.Error) {
285
281
  next(error);
286
282
  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
283
  }
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;
284
+ // TODO: Determine the correct error scenario.
285
+ req.config.loggerController.error('error getting auth for sessionToken', error);
286
+ throw new _node.default.Error(_node.default.Error.UNKNOWN_ERROR, error);
311
287
  }
288
+ };
289
+ exports.handleParseSession = handleParseSession;
290
+ function getClientIp(req) {
291
+ return req.ip;
312
292
  }
313
-
314
293
  function httpAuth(req) {
315
294
  if (!(req.req || req).headers.authorization) return;
316
295
  var header = (req.req || req).headers.authorization;
317
- var appId, masterKey, javascriptKey; // parse header
296
+ var appId, masterKey, javascriptKey;
318
297
 
298
+ // parse header
319
299
  var authPrefix = 'basic ';
320
300
  var match = header.toLowerCase().indexOf(authPrefix);
321
-
322
301
  if (match == 0) {
323
302
  var encodedAuth = header.substring(authPrefix.length, header.length);
324
303
  var credentials = decodeBase64(encodedAuth).split(':');
325
-
326
304
  if (credentials.length == 2) {
327
305
  appId = credentials[0];
328
306
  var key = credentials[1];
329
307
  var jsKeyPrefix = 'javascript-key=';
330
308
  var matchKey = key.indexOf(jsKeyPrefix);
331
-
332
309
  if (matchKey == 0) {
333
310
  javascriptKey = key.substring(jsKeyPrefix.length, key.length);
334
311
  } else {
@@ -336,34 +313,28 @@ function httpAuth(req) {
336
313
  }
337
314
  }
338
315
  }
339
-
340
316
  return {
341
317
  appId: appId,
342
318
  masterKey: masterKey,
343
319
  javascriptKey: javascriptKey
344
320
  };
345
321
  }
346
-
347
322
  function decodeBase64(str) {
348
323
  return Buffer.from(str, 'base64').toString();
349
324
  }
350
-
351
325
  function allowCrossDomain(appId) {
352
326
  return (req, res, next) => {
353
327
  const config = _Config.default.get(appId, getMountForRequest(req));
354
-
355
328
  let allowHeaders = DEFAULT_ALLOWED_HEADERS;
356
-
357
329
  if (config && config.allowHeaders) {
358
330
  allowHeaders += `, ${config.allowHeaders.join(', ')}`;
359
331
  }
360
-
361
332
  const allowOrigin = config && config.allowOrigin || '*';
362
333
  res.header('Access-Control-Allow-Origin', allowOrigin);
363
334
  res.header('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE,OPTIONS');
364
335
  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
-
336
+ res.header('Access-Control-Expose-Headers', 'X-Parse-Job-Status-Id, X-Parse-Push-Status-Id');
337
+ // intercept OPTIONS method
367
338
  if ('OPTIONS' == req.method) {
368
339
  res.sendStatus(200);
369
340
  } else {
@@ -371,40 +342,32 @@ function allowCrossDomain(appId) {
371
342
  }
372
343
  };
373
344
  }
374
-
375
345
  function allowMethodOverride(req, res, next) {
376
346
  if (req.method === 'POST' && req.body._method) {
377
347
  req.originalMethod = req.method;
378
348
  req.method = req.body._method;
379
349
  delete req.body._method;
380
350
  }
381
-
382
351
  next();
383
352
  }
384
-
385
353
  function handleParseErrors(err, req, res, next) {
386
354
  const log = req.config && req.config.loggerController || _logger.default;
387
-
388
355
  if (err instanceof _node.default.Error) {
389
356
  if (req.config && req.config.enableExpressErrorHandler) {
390
357
  return next(err);
391
358
  }
392
-
393
- let httpStatus; // TODO: fill out this mapping
394
-
359
+ let httpStatus;
360
+ // TODO: fill out this mapping
395
361
  switch (err.code) {
396
362
  case _node.default.Error.INTERNAL_SERVER_ERROR:
397
363
  httpStatus = 500;
398
364
  break;
399
-
400
365
  case _node.default.Error.OBJECT_NOT_FOUND:
401
366
  httpStatus = 404;
402
367
  break;
403
-
404
368
  default:
405
369
  httpStatus = 400;
406
370
  }
407
-
408
371
  res.status(httpStatus);
409
372
  res.json({
410
373
  code: err.code,
@@ -416,7 +379,6 @@ function handleParseErrors(err, req, res, next) {
416
379
  res.json({
417
380
  error: err.message
418
381
  });
419
-
420
382
  if (!(process && process.env.TESTING)) {
421
383
  next(err);
422
384
  }
@@ -427,23 +389,19 @@ function handleParseErrors(err, req, res, next) {
427
389
  code: _node.default.Error.INTERNAL_SERVER_ERROR,
428
390
  message: 'Internal server error.'
429
391
  });
430
-
431
392
  if (!(process && process.env.TESTING)) {
432
393
  next(err);
433
394
  }
434
395
  }
435
396
  }
436
-
437
397
  function enforceMasterKeyAccess(req, res, next) {
438
398
  if (!req.auth.isMaster) {
439
399
  res.status(403);
440
400
  res.end('{"error":"unauthorized: master key is required"}');
441
401
  return;
442
402
  }
443
-
444
403
  next();
445
404
  }
446
-
447
405
  function promiseEnforceMasterKeyAccess(request) {
448
406
  if (!request.auth.isMaster) {
449
407
  const error = new Error();
@@ -451,56 +409,97 @@ function promiseEnforceMasterKeyAccess(request) {
451
409
  error.message = 'unauthorized: master key is required';
452
410
  throw error;
453
411
  }
454
-
455
412
  return Promise.resolve();
456
413
  }
414
+ const addRateLimit = (route, config) => {
415
+ if (typeof config === 'string') {
416
+ config = _Config.default.get(config);
417
+ }
418
+ for (const key in route) {
419
+ if (!_Definitions.RateLimitOptions[key]) {
420
+ throw `Invalid rate limit option "${key}"`;
421
+ }
422
+ }
423
+ if (!config.rateLimits) {
424
+ config.rateLimits = [];
425
+ }
426
+ config.rateLimits.push({
427
+ path: (0, _pathToRegexp.default)(route.requestPath),
428
+ handler: (0, _expressRateLimit.default)({
429
+ windowMs: route.requestTimeWindow,
430
+ max: route.requestCount,
431
+ message: route.errorResponseMessage || _Definitions.RateLimitOptions.errorResponseMessage.default,
432
+ handler: (request, response, next, options) => {
433
+ throw options.message;
434
+ },
435
+ skip: request => {
436
+ if (request.ip === '127.0.0.1' && !route.includeInternalRequests) {
437
+ return true;
438
+ }
439
+ if (route.includeMasterKey) {
440
+ return false;
441
+ }
442
+ if (route.requestMethods) {
443
+ if (Array.isArray(route.requestMethods)) {
444
+ if (!route.requestMethods.includes(request.method)) {
445
+ return true;
446
+ }
447
+ } else {
448
+ const regExp = new RegExp(route.requestMethods);
449
+ if (!regExp.test(request.method)) {
450
+ return true;
451
+ }
452
+ }
453
+ }
454
+ return request.auth.isMaster;
455
+ },
456
+ keyGenerator: request => {
457
+ return request.config.ip;
458
+ }
459
+ })
460
+ });
461
+ _Config.default.put(config);
462
+ };
463
+
457
464
  /**
458
465
  * Deduplicates a request to ensure idempotency. Duplicates are determined by the request ID
459
466
  * in the request header. If a request has no request ID, it is executed anyway.
460
467
  * @param {*} req The request to evaluate.
461
468
  * @returns Promise<{}>
462
469
  */
463
-
464
-
470
+ exports.addRateLimit = addRateLimit;
465
471
  function promiseEnsureIdempotency(req) {
466
472
  // Enable feature only for MongoDB
467
473
  if (!(req.config.database.adapter instanceof _MongoStorageAdapter.default || req.config.database.adapter instanceof _PostgresStorageAdapter.default)) {
468
474
  return Promise.resolve();
469
- } // Get parameters
470
-
471
-
475
+ }
476
+ // Get parameters
472
477
  const config = req.config;
473
478
  const requestId = ((req || {}).headers || {})['x-parse-request-id'];
474
479
  const {
475
480
  paths,
476
481
  ttl
477
482
  } = config.idempotencyOptions;
478
-
479
483
  if (!requestId || !config.idempotencyOptions) {
480
484
  return Promise.resolve();
481
- } // Request path may contain trailing slashes, depending on the original request, so remove
485
+ }
486
+ // Request path may contain trailing slashes, depending on the original request, so remove
482
487
  // 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
-
488
+ const reqPath = req.path.replace(/^\/|\/$/, '');
489
+ // Determine whether idempotency is enabled for current request path
487
490
  let match = false;
488
-
489
491
  for (const path of paths) {
490
492
  // Assume one wants a path to always match from the beginning to prevent any mistakes
491
493
  const regex = new RegExp(path.charAt(0) === '^' ? path : '^' + path);
492
-
493
494
  if (reqPath.match(regex)) {
494
495
  match = true;
495
496
  break;
496
497
  }
497
498
  }
498
-
499
499
  if (!match) {
500
500
  return Promise.resolve();
501
- } // Try to store request
502
-
503
-
501
+ }
502
+ // Try to store request
504
503
  const expiryDate = new Date(new Date().setSeconds(new Date().getSeconds() + ttl));
505
504
  return _rest.default.create(config, _Auth.default.master(config), '_Idempotency', {
506
505
  reqId: requestId,
@@ -509,16 +508,13 @@ function promiseEnsureIdempotency(req) {
509
508
  if (e.code == _node.default.Error.DUPLICATE_VALUE) {
510
509
  throw new _node.default.Error(_node.default.Error.DUPLICATE_REQUEST, 'Duplicate request');
511
510
  }
512
-
513
511
  throw e;
514
512
  });
515
513
  }
516
-
517
514
  function invalidRequest(req, res) {
518
515
  res.status(403);
519
516
  res.end('{"error":"unauthorized"}');
520
517
  }
521
-
522
518
  function malformedContext(req, res) {
523
519
  res.status(400);
524
520
  res.json({
@@ -526,4 +522,4 @@ function malformedContext(req, res) {
526
522
  error: 'Invalid object for context.'
527
523
  });
528
524
  }
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":";;;;;;;;;;;;;;AAAA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;;;AAEO,MAAMA,uBAAuB,GAClC,+OADK;;;AAGP,MAAMC,kBAAkB,GAAG,UAAUC,GAAV,EAAe;AACxC,QAAMC,eAAe,GAAGD,GAAG,CAACE,WAAJ,CAAgBC,MAAhB,GAAyBH,GAAG,CAACI,GAAJ,CAAQD,MAAzD;AACA,QAAME,SAAS,GAAGL,GAAG,CAACE,WAAJ,CAAgBI,KAAhB,CAAsB,CAAtB,EAAyBL,eAAzB,CAAlB;AACA,SAAOD,GAAG,CAACO,QAAJ,GAAe,KAAf,GAAuBP,GAAG,CAACQ,GAAJ,CAAQ,MAAR,CAAvB,GAAyCH,SAAhD;AACD,CAJD,C,CAMA;AACA;AACA;AACA;AACA;AACA;;;AACO,SAASI,kBAAT,CAA4BT,GAA5B,EAAiCU,GAAjC,EAAsCC,IAAtC,EAA4C;AACjD,MAAIC,KAAK,GAAGb,kBAAkB,CAACC,GAAD,CAA9B;AAEA,MAAIa,OAAO,GAAG,EAAd;;AACA,MAAIb,GAAG,CAACQ,GAAJ,CAAQ,uBAAR,KAAoC,IAAxC,EAA8C;AAC5C,QAAI;AACFK,MAAAA,OAAO,GAAGC,IAAI,CAACC,KAAL,CAAWf,GAAG,CAACQ,GAAJ,CAAQ,uBAAR,CAAX,CAAV;;AACA,UAAIQ,MAAM,CAACC,SAAP,CAAiBC,QAAjB,CAA0BC,IAA1B,CAA+BN,OAA/B,MAA4C,iBAAhD,EAAmE;AACjE,cAAM,0BAAN;AACD;AACF,KALD,CAKE,OAAOO,CAAP,EAAU;AACV,aAAOC,gBAAgB,CAACrB,GAAD,EAAMU,GAAN,CAAvB;AACD;AACF;;AACD,MAAIY,IAAI,GAAG;AACTC,IAAAA,KAAK,EAAEvB,GAAG,CAACQ,GAAJ,CAAQ,wBAAR,CADE;AAETgB,IAAAA,YAAY,EAAExB,GAAG,CAACQ,GAAJ,CAAQ,uBAAR,CAFL;AAGTiB,IAAAA,SAAS,EAAEzB,GAAG,CAACQ,GAAJ,CAAQ,oBAAR,CAHF;AAITkB,IAAAA,cAAc,EAAE1B,GAAG,CAACQ,GAAJ,CAAQ,yBAAR,CAJP;AAKTmB,IAAAA,SAAS,EAAE3B,GAAG,CAACQ,GAAJ,CAAQ,oBAAR,CALF;AAMToB,IAAAA,aAAa,EAAE5B,GAAG,CAACQ,GAAJ,CAAQ,wBAAR,CANN;AAOTqB,IAAAA,SAAS,EAAE7B,GAAG,CAACQ,GAAJ,CAAQ,qBAAR,CAPF;AAQTsB,IAAAA,UAAU,EAAE9B,GAAG,CAACQ,GAAJ,CAAQ,sBAAR,CARH;AASTuB,IAAAA,aAAa,EAAE/B,GAAG,CAACQ,GAAJ,CAAQ,wBAAR,CATN;AAUTK,IAAAA,OAAO,EAAEA;AAVA,GAAX;AAaA,MAAImB,SAAS,GAAGC,QAAQ,CAACjC,GAAD,CAAxB;;AAEA,MAAIgC,SAAJ,EAAe;AACb,QAAIE,cAAc,GAAGF,SAAS,CAACT,KAA/B;;AACA,QAAIY,eAAS3B,GAAT,CAAa0B,cAAb,CAAJ,EAAkC;AAChCZ,MAAAA,IAAI,CAACC,KAAL,GAAaW,cAAb;AACAZ,MAAAA,IAAI,CAACG,SAAL,GAAiBO,SAAS,CAACP,SAAV,IAAuBH,IAAI,CAACG,SAA7C;AACAH,MAAAA,IAAI,CAACM,aAAL,GAAqBI,SAAS,CAACJ,aAAV,IAA2BN,IAAI,CAACM,aAArD;AACD;AACF;;AAED,MAAI5B,GAAG,CAACoC,IAAR,EAAc;AACZ;AACA;AACA,WAAOpC,GAAG,CAACoC,IAAJ,CAASC,OAAhB;AACD;;AAED,MAAIC,WAAW,GAAG,KAAlB;;AAEA,MAAI,CAAChB,IAAI,CAACC,KAAN,IAAe,CAACY,eAAS3B,GAAT,CAAac,IAAI,CAACC,KAAlB,CAApB,EAA8C;AAC5C;AACA,QAAIvB,GAAG,CAACoC,IAAJ,YAAoBG,MAAxB,EAAgC;AAC9B;AACA;AACA;AACA;AACA;AACA,UAAI;AACFvC,QAAAA,GAAG,CAACoC,IAAJ,GAAWtB,IAAI,CAACC,KAAL,CAAWf,GAAG,CAACoC,IAAf,CAAX;AACD,OAFD,CAEE,OAAOhB,CAAP,EAAU;AACV,eAAOoB,cAAc,CAACxC,GAAD,EAAMU,GAAN,CAArB;AACD;;AACD4B,MAAAA,WAAW,GAAG,IAAd;AACD;;AAED,QAAItC,GAAG,CAACoC,IAAR,EAAc;AACZ,aAAOpC,GAAG,CAACoC,IAAJ,CAASK,iBAAhB;AACD;;AAED,QACEzC,GAAG,CAACoC,IAAJ,IACApC,GAAG,CAACoC,IAAJ,CAASM,cADT,IAEAP,eAAS3B,GAAT,CAAaR,GAAG,CAACoC,IAAJ,CAASM,cAAtB,CAFA,KAGC,CAACpB,IAAI,CAACG,SAAN,IAAmBU,eAAS3B,GAAT,CAAaR,GAAG,CAACoC,IAAJ,CAASM,cAAtB,EAAsCjB,SAAtC,KAAoDH,IAAI,CAACG,SAH7E,CADF,EAKE;AACAH,MAAAA,IAAI,CAACC,KAAL,GAAavB,GAAG,CAACoC,IAAJ,CAASM,cAAtB;AACApB,MAAAA,IAAI,CAACM,aAAL,GAAqB5B,GAAG,CAACoC,IAAJ,CAASO,cAAT,IAA2B,EAAhD;AACA,aAAO3C,GAAG,CAACoC,IAAJ,CAASM,cAAhB;AACA,aAAO1C,GAAG,CAACoC,IAAJ,CAASO,cAAhB,CAJA,CAKA;AACA;;AACA,UAAI3C,GAAG,CAACoC,IAAJ,CAASQ,cAAb,EAA6B;AAC3BtB,QAAAA,IAAI,CAACS,aAAL,GAAqB/B,GAAG,CAACoC,IAAJ,CAASQ,cAA9B;AACA,eAAO5C,GAAG,CAACoC,IAAJ,CAASQ,cAAhB;AACD;;AACD,UAAI5C,GAAG,CAACoC,IAAJ,CAASS,eAAb,EAA8B;AAC5BvB,QAAAA,IAAI,CAACI,cAAL,GAAsB1B,GAAG,CAACoC,IAAJ,CAASS,eAA/B;AACA,eAAO7C,GAAG,CAACoC,IAAJ,CAASS,eAAhB;AACD;;AACD,UAAI7C,GAAG,CAACoC,IAAJ,CAASU,aAAb,EAA4B;AAC1BxB,QAAAA,IAAI,CAACE,YAAL,GAAoBxB,GAAG,CAACoC,IAAJ,CAASU,aAA7B;AACA,eAAO9C,GAAG,CAACoC,IAAJ,CAASU,aAAhB;AACD;;AACD,UAAI9C,GAAG,CAACoC,IAAJ,CAASW,UAAb,EAAyB;AACvBzB,QAAAA,IAAI,CAACG,SAAL,GAAiBzB,GAAG,CAACoC,IAAJ,CAASW,UAA1B;AACA,eAAO/C,GAAG,CAACoC,IAAJ,CAASW,UAAhB;AACD;;AACD,UAAI/C,GAAG,CAACoC,IAAJ,CAASY,QAAb,EAAuB;AACrB,YAAIhD,GAAG,CAACoC,IAAJ,CAASY,QAAT,YAA6BhC,MAAjC,EAAyC;AACvCM,UAAAA,IAAI,CAACT,OAAL,GAAeb,GAAG,CAACoC,IAAJ,CAASY,QAAxB;AACD,SAFD,MAEO;AACL,cAAI;AACF1B,YAAAA,IAAI,CAACT,OAAL,GAAeC,IAAI,CAACC,KAAL,CAAWf,GAAG,CAACoC,IAAJ,CAASY,QAApB,CAAf;;AACA,gBAAIhC,MAAM,CAACC,SAAP,CAAiBC,QAAjB,CAA0BC,IAA1B,CAA+BG,IAAI,CAACT,OAApC,MAAiD,iBAArD,EAAwE;AACtE,oBAAM,0BAAN;AACD;AACF,WALD,CAKE,OAAOO,CAAP,EAAU;AACV,mBAAOC,gBAAgB,CAACrB,GAAD,EAAMU,GAAN,CAAvB;AACD;AACF;;AACD,eAAOV,GAAG,CAACoC,IAAJ,CAASY,QAAhB;AACD;;AACD,UAAIhD,GAAG,CAACoC,IAAJ,CAASa,YAAb,EAA2B;AACzBjD,QAAAA,GAAG,CAACkD,OAAJ,CAAY,cAAZ,IAA8BlD,GAAG,CAACoC,IAAJ,CAASa,YAAvC;AACA,eAAOjD,GAAG,CAACoC,IAAJ,CAASa,YAAhB;AACD;AACF,KA/CD,MA+CO;AACL,aAAOT,cAAc,CAACxC,GAAD,EAAMU,GAAN,CAArB;AACD;AACF;;AAED,MAAIY,IAAI,CAACE,YAAL,IAAqB,OAAOF,IAAI,CAACE,YAAZ,KAA6B,QAAtD,EAAgE;AAC9DF,IAAAA,IAAI,CAACE,YAAL,GAAoBF,IAAI,CAACE,YAAL,CAAkBN,QAAlB,EAApB;AACD;;AAED,MAAII,IAAI,CAACS,aAAT,EAAwB;AACtBT,IAAAA,IAAI,CAAC6B,SAAL,GAAiBC,mBAAUC,UAAV,CAAqB/B,IAAI,CAACS,aAA1B,CAAjB;AACD;;AAED,MAAIO,WAAJ,EAAiB;AACftC,IAAAA,GAAG,CAACsD,QAAJ,GAAetD,GAAG,CAACoC,IAAJ,CAASkB,QAAxB,CADe,CAEf;;AACA,QAAIC,MAAM,GAAGvD,GAAG,CAACoC,IAAJ,CAASmB,MAAtB;AACAvD,IAAAA,GAAG,CAACoC,IAAJ,GAAWG,MAAM,CAACiB,IAAP,CAAYD,MAAZ,EAAoB,QAApB,CAAX;AACD;;AAED,QAAME,QAAQ,GAAGC,WAAW,CAAC1D,GAAD,CAA5B;AAEAsB,EAAAA,IAAI,CAACqC,GAAL,GAAWxB,eAAS3B,GAAT,CAAac,IAAI,CAACC,KAAlB,CAAX;AACAvB,EAAAA,GAAG,CAAC4D,MAAJ,GAAaC,gBAAOrD,GAAP,CAAWc,IAAI,CAACC,KAAhB,EAAuBX,KAAvB,CAAb;AACAZ,EAAAA,GAAG,CAAC4D,MAAJ,CAAWV,OAAX,GAAqBlD,GAAG,CAACkD,OAAJ,IAAe,EAApC;AACAlD,EAAAA,GAAG,CAAC4D,MAAJ,CAAWE,EAAX,GAAgBL,QAAhB;AACAzD,EAAAA,GAAG,CAACsB,IAAJ,GAAWA,IAAX;;AAEA,MACEA,IAAI,CAACG,SAAL,IACAzB,GAAG,CAAC4D,MAAJ,CAAWG,YADX,IAEA/D,GAAG,CAAC4D,MAAJ,CAAWG,YAAX,CAAwB5D,MAAxB,KAAmC,CAFnC,IAGAH,GAAG,CAAC4D,MAAJ,CAAWG,YAAX,CAAwBC,OAAxB,CAAgCP,QAAhC,MAA8C,CAAC,CAJjD,EAKE;AACA,WAAOjB,cAAc,CAACxC,GAAD,EAAMU,GAAN,CAArB;AACD;;AAED,MAAIuD,QAAQ,GAAG3C,IAAI,CAACG,SAAL,KAAmBzB,GAAG,CAAC4D,MAAJ,CAAWnC,SAA7C;;AAEA,MAAIwC,QAAJ,EAAc;AACZjE,IAAAA,GAAG,CAACkE,IAAJ,GAAW,IAAIA,cAAKC,IAAT,CAAc;AACvBP,MAAAA,MAAM,EAAE5D,GAAG,CAAC4D,MADW;AAEvBlC,MAAAA,cAAc,EAAEJ,IAAI,CAACI,cAFE;AAGvBuC,MAAAA,QAAQ,EAAE;AAHa,KAAd,CAAX;AAKAtD,IAAAA,IAAI;AACJ;AACD;;AAED,MAAIyD,gBAAgB,GAAG9C,IAAI,CAACG,SAAL,KAAmBzB,GAAG,CAAC4D,MAAJ,CAAWS,iBAArD;;AACA,MACE,OAAOrE,GAAG,CAAC4D,MAAJ,CAAWS,iBAAlB,IAAuC,WAAvC,IACArE,GAAG,CAAC4D,MAAJ,CAAWS,iBADX,IAEAD,gBAHF,EAIE;AACApE,IAAAA,GAAG,CAACkE,IAAJ,GAAW,IAAIA,cAAKC,IAAT,CAAc;AACvBP,MAAAA,MAAM,EAAE5D,GAAG,CAAC4D,MADW;AAEvBlC,MAAAA,cAAc,EAAEJ,IAAI,CAACI,cAFE;AAGvBuC,MAAAA,QAAQ,EAAE,IAHa;AAIvBK,MAAAA,UAAU,EAAE;AAJW,KAAd,CAAX;AAMA3D,IAAAA,IAAI;AACJ;AACD,GAhLgD,CAkLjD;AACA;;;AACA,QAAM4D,IAAI,GAAG,CAAC,WAAD,EAAc,eAAd,EAA+B,WAA/B,EAA4C,YAA5C,CAAb;AACA,QAAMC,gBAAgB,GAAGD,IAAI,CAACE,IAAL,CAAU,UAAUC,GAAV,EAAe;AAChD,WAAO1E,GAAG,CAAC4D,MAAJ,CAAWc,GAAX,MAAoBC,SAA3B;AACD,GAFwB,CAAzB;AAGA,QAAMC,aAAa,GAAGL,IAAI,CAACE,IAAL,CAAU,UAAUC,GAAV,EAAe;AAC7C,WAAO1E,GAAG,CAAC4D,MAAJ,CAAWc,GAAX,MAAoBC,SAApB,IAAiCrD,IAAI,CAACoD,GAAD,CAAJ,KAAc1E,GAAG,CAAC4D,MAAJ,CAAWc,GAAX,CAAtD;AACD,GAFqB,CAAtB;;AAIA,MAAIF,gBAAgB,IAAI,CAACI,aAAzB,EAAwC;AACtC,WAAOpC,cAAc,CAACxC,GAAD,EAAMU,GAAN,CAArB;AACD;;AAED,MAAIV,GAAG,CAACI,GAAJ,IAAW,QAAf,EAAyB;AACvB,WAAOkB,IAAI,CAACE,YAAZ;AACD;;AAED,MAAIxB,GAAG,CAAC6E,WAAR,EAAqB;AACnB7E,IAAAA,GAAG,CAACkE,IAAJ,GAAW,IAAIA,cAAKC,IAAT,CAAc;AACvBP,MAAAA,MAAM,EAAE5D,GAAG,CAAC4D,MADW;AAEvBlC,MAAAA,cAAc,EAAEJ,IAAI,CAACI,cAFE;AAGvBuC,MAAAA,QAAQ,EAAE,KAHa;AAIvBa,MAAAA,IAAI,EAAE9E,GAAG,CAAC6E;AAJa,KAAd,CAAX;AAMAlE,IAAAA,IAAI;AACJ;AACD;;AAED,MAAI,CAACW,IAAI,CAACE,YAAV,EAAwB;AACtBxB,IAAAA,GAAG,CAACkE,IAAJ,GAAW,IAAIA,cAAKC,IAAT,CAAc;AACvBP,MAAAA,MAAM,EAAE5D,GAAG,CAAC4D,MADW;AAEvBlC,MAAAA,cAAc,EAAEJ,IAAI,CAACI,cAFE;AAGvBuC,MAAAA,QAAQ,EAAE;AAHa,KAAd,CAAX;AAKAtD,IAAAA,IAAI;AACJ;AACD;;AAED,SAAOoE,OAAO,CAACC,OAAR,GACJC,IADI,CACC,MAAM;AACV;AACA,QACE3D,IAAI,CAACE,YAAL,IACAxB,GAAG,CAACI,GAAJ,KAAY,4BADZ,IAEAkB,IAAI,CAACE,YAAL,CAAkBwC,OAAlB,CAA0B,IAA1B,KAAmC,CAHrC,EAIE;AACA,aAAOE,cAAKgB,4BAAL,CAAkC;AACvCtB,QAAAA,MAAM,EAAE5D,GAAG,CAAC4D,MAD2B;AAEvClC,QAAAA,cAAc,EAAEJ,IAAI,CAACI,cAFkB;AAGvCF,QAAAA,YAAY,EAAEF,IAAI,CAACE;AAHoB,OAAlC,CAAP;AAKD,KAVD,MAUO;AACL,aAAO0C,cAAKiB,sBAAL,CAA4B;AACjCvB,QAAAA,MAAM,EAAE5D,GAAG,CAAC4D,MADqB;AAEjClC,QAAAA,cAAc,EAAEJ,IAAI,CAACI,cAFY;AAGjCF,QAAAA,YAAY,EAAEF,IAAI,CAACE;AAHc,OAA5B,CAAP;AAKD;AACF,GApBI,EAqBJyD,IArBI,CAqBCf,IAAI,IAAI;AACZ,QAAIA,IAAJ,EAAU;AACRlE,MAAAA,GAAG,CAACkE,IAAJ,GAAWA,IAAX;AACAvD,MAAAA,IAAI;AACL;AACF,GA1BI,EA2BJyE,KA3BI,CA2BEC,KAAK,IAAI;AACd,QAAIA,KAAK,YAAYC,cAAMC,KAA3B,EAAkC;AAChC5E,MAAAA,IAAI,CAAC0E,KAAD,CAAJ;AACA;AACD,KAHD,MAGO;AACL;AACArF,MAAAA,GAAG,CAAC4D,MAAJ,CAAW4B,gBAAX,CAA4BH,KAA5B,CAAkC,qCAAlC,EAAyEA,KAAzE;AACA,YAAM,IAAIC,cAAMC,KAAV,CAAgBD,cAAMC,KAAN,CAAYE,aAA5B,EAA2CJ,KAA3C,CAAN;AACD;AACF,GApCI,CAAP;AAqCD;;AAED,SAAS3B,WAAT,CAAqB1D,GAArB,EAA0B;AACxB,MAAIA,GAAG,CAACkD,OAAJ,CAAY,iBAAZ,CAAJ,EAAoC;AAClC;AACA,WAAOlD,GAAG,CAACkD,OAAJ,CAAY,iBAAZ,EAA+BwC,KAA/B,CAAqC,GAArC,EAA0C,CAA1C,CAAP;AACD,GAHD,MAGO,IAAI1F,GAAG,CAAC2F,UAAJ,IAAkB3F,GAAG,CAAC2F,UAAJ,CAAeC,aAArC,EAAoD;AACzD;AACA,WAAO5F,GAAG,CAAC2F,UAAJ,CAAeC,aAAtB;AACD,GAHM,MAGA,IAAI5F,GAAG,CAAC6F,MAAR,EAAgB;AACrB;AACA,WAAO7F,GAAG,CAAC6F,MAAJ,CAAWD,aAAlB;AACD,GAHM,MAGA,IAAI5F,GAAG,CAAC2F,UAAJ,IAAkB3F,GAAG,CAAC2F,UAAJ,CAAeE,MAArC,EAA6C;AAClD;AACA,WAAO7F,GAAG,CAAC2F,UAAJ,CAAeE,MAAf,CAAsBD,aAA7B;AACD,GAHM,MAGA;AACL;AACA,WAAO5F,GAAG,CAAC8D,EAAX;AACD;AACF;;AAED,SAAS7B,QAAT,CAAkBjC,GAAlB,EAAuB;AACrB,MAAI,CAAC,CAACA,GAAG,CAACA,GAAJ,IAAWA,GAAZ,EAAiBkD,OAAjB,CAAyB4C,aAA9B,EAA6C;AAE7C,MAAIC,MAAM,GAAG,CAAC/F,GAAG,CAACA,GAAJ,IAAWA,GAAZ,EAAiBkD,OAAjB,CAAyB4C,aAAtC;AACA,MAAIvE,KAAJ,EAAWE,SAAX,EAAsBG,aAAtB,CAJqB,CAMrB;;AACA,MAAIoE,UAAU,GAAG,QAAjB;AAEA,MAAIC,KAAK,GAAGF,MAAM,CAACG,WAAP,GAAqBlC,OAArB,CAA6BgC,UAA7B,CAAZ;;AAEA,MAAIC,KAAK,IAAI,CAAb,EAAgB;AACd,QAAIE,WAAW,GAAGJ,MAAM,CAACK,SAAP,CAAiBJ,UAAU,CAAC7F,MAA5B,EAAoC4F,MAAM,CAAC5F,MAA3C,CAAlB;AACA,QAAIkG,WAAW,GAAGC,YAAY,CAACH,WAAD,CAAZ,CAA0BT,KAA1B,CAAgC,GAAhC,CAAlB;;AAEA,QAAIW,WAAW,CAAClG,MAAZ,IAAsB,CAA1B,EAA6B;AAC3BoB,MAAAA,KAAK,GAAG8E,WAAW,CAAC,CAAD,CAAnB;AACA,UAAI3B,GAAG,GAAG2B,WAAW,CAAC,CAAD,CAArB;AAEA,UAAIE,WAAW,GAAG,iBAAlB;AAEA,UAAIC,QAAQ,GAAG9B,GAAG,CAACV,OAAJ,CAAYuC,WAAZ,CAAf;;AACA,UAAIC,QAAQ,IAAI,CAAhB,EAAmB;AACjB5E,QAAAA,aAAa,GAAG8C,GAAG,CAAC0B,SAAJ,CAAcG,WAAW,CAACpG,MAA1B,EAAkCuE,GAAG,CAACvE,MAAtC,CAAhB;AACD,OAFD,MAEO;AACLsB,QAAAA,SAAS,GAAGiD,GAAZ;AACD;AACF;AACF;;AAED,SAAO;AAAEnD,IAAAA,KAAK,EAAEA,KAAT;AAAgBE,IAAAA,SAAS,EAAEA,SAA3B;AAAsCG,IAAAA,aAAa,EAAEA;AAArD,GAAP;AACD;;AAED,SAAS0E,YAAT,CAAsBG,GAAtB,EAA2B;AACzB,SAAOlE,MAAM,CAACiB,IAAP,CAAYiD,GAAZ,EAAiB,QAAjB,EAA2BvF,QAA3B,EAAP;AACD;;AAEM,SAASwF,gBAAT,CAA0BnF,KAA1B,EAAiC;AACtC,SAAO,CAACvB,GAAD,EAAMU,GAAN,EAAWC,IAAX,KAAoB;AACzB,UAAMiD,MAAM,GAAGC,gBAAOrD,GAAP,CAAWe,KAAX,EAAkBxB,kBAAkB,CAACC,GAAD,CAApC,CAAf;;AACA,QAAI2G,YAAY,GAAG7G,uBAAnB;;AACA,QAAI8D,MAAM,IAAIA,MAAM,CAAC+C,YAArB,EAAmC;AACjCA,MAAAA,YAAY,IAAK,KAAI/C,MAAM,CAAC+C,YAAP,CAAoBC,IAApB,CAAyB,IAAzB,CAA+B,EAApD;AACD;;AACD,UAAMC,WAAW,GAAIjD,MAAM,IAAIA,MAAM,CAACiD,WAAlB,IAAkC,GAAtD;AACAnG,IAAAA,GAAG,CAACqF,MAAJ,CAAW,6BAAX,EAA0Cc,WAA1C;AACAnG,IAAAA,GAAG,CAACqF,MAAJ,CAAW,8BAAX,EAA2C,6BAA3C;AACArF,IAAAA,GAAG,CAACqF,MAAJ,CAAW,8BAAX,EAA2CY,YAA3C;AACAjG,IAAAA,GAAG,CAACqF,MAAJ,CAAW,+BAAX,EAA4C,+CAA5C,EAVyB,CAWzB;;AACA,QAAI,aAAa/F,GAAG,CAAC8G,MAArB,EAA6B;AAC3BpG,MAAAA,GAAG,CAACqG,UAAJ,CAAe,GAAf;AACD,KAFD,MAEO;AACLpG,MAAAA,IAAI;AACL;AACF,GAjBD;AAkBD;;AAEM,SAASqG,mBAAT,CAA6BhH,GAA7B,EAAkCU,GAAlC,EAAuCC,IAAvC,EAA6C;AAClD,MAAIX,GAAG,CAAC8G,MAAJ,KAAe,MAAf,IAAyB9G,GAAG,CAACoC,IAAJ,CAAS6E,OAAtC,EAA+C;AAC7CjH,IAAAA,GAAG,CAACkH,cAAJ,GAAqBlH,GAAG,CAAC8G,MAAzB;AACA9G,IAAAA,GAAG,CAAC8G,MAAJ,GAAa9G,GAAG,CAACoC,IAAJ,CAAS6E,OAAtB;AACA,WAAOjH,GAAG,CAACoC,IAAJ,CAAS6E,OAAhB;AACD;;AACDtG,EAAAA,IAAI;AACL;;AAEM,SAASwG,iBAAT,CAA2BC,GAA3B,EAAgCpH,GAAhC,EAAqCU,GAArC,EAA0CC,IAA1C,EAAgD;AACrD,QAAM0G,GAAG,GAAIrH,GAAG,CAAC4D,MAAJ,IAAc5D,GAAG,CAAC4D,MAAJ,CAAW4B,gBAA1B,IAA+C8B,eAA3D;;AACA,MAAIF,GAAG,YAAY9B,cAAMC,KAAzB,EAAgC;AAC9B,QAAIvF,GAAG,CAAC4D,MAAJ,IAAc5D,GAAG,CAAC4D,MAAJ,CAAW2D,yBAA7B,EAAwD;AACtD,aAAO5G,IAAI,CAACyG,GAAD,CAAX;AACD;;AACD,QAAII,UAAJ,CAJ8B,CAK9B;;AACA,YAAQJ,GAAG,CAACK,IAAZ;AACE,WAAKnC,cAAMC,KAAN,CAAYmC,qBAAjB;AACEF,QAAAA,UAAU,GAAG,GAAb;AACA;;AACF,WAAKlC,cAAMC,KAAN,CAAYoC,gBAAjB;AACEH,QAAAA,UAAU,GAAG,GAAb;AACA;;AACF;AACEA,QAAAA,UAAU,GAAG,GAAb;AARJ;;AAUA9G,IAAAA,GAAG,CAACkH,MAAJ,CAAWJ,UAAX;AACA9G,IAAAA,GAAG,CAACmH,IAAJ,CAAS;AAAEJ,MAAAA,IAAI,EAAEL,GAAG,CAACK,IAAZ;AAAkBpC,MAAAA,KAAK,EAAE+B,GAAG,CAACU;AAA7B,KAAT;AACAT,IAAAA,GAAG,CAAChC,KAAJ,CAAU,eAAV,EAA2B+B,GAA3B;AACD,GAnBD,MAmBO,IAAIA,GAAG,CAACQ,MAAJ,IAAcR,GAAG,CAACU,OAAtB,EAA+B;AACpCpH,IAAAA,GAAG,CAACkH,MAAJ,CAAWR,GAAG,CAACQ,MAAf;AACAlH,IAAAA,GAAG,CAACmH,IAAJ,CAAS;AAAExC,MAAAA,KAAK,EAAE+B,GAAG,CAACU;AAAb,KAAT;;AACA,QAAI,EAAEC,OAAO,IAAIA,OAAO,CAACC,GAAR,CAAYC,OAAzB,CAAJ,EAAuC;AACrCtH,MAAAA,IAAI,CAACyG,GAAD,CAAJ;AACD;AACF,GANM,MAMA;AACLC,IAAAA,GAAG,CAAChC,KAAJ,CAAU,iCAAV,EAA6C+B,GAA7C,EAAkDA,GAAG,CAACc,KAAtD;AACAxH,IAAAA,GAAG,CAACkH,MAAJ,CAAW,GAAX;AACAlH,IAAAA,GAAG,CAACmH,IAAJ,CAAS;AACPJ,MAAAA,IAAI,EAAEnC,cAAMC,KAAN,CAAYmC,qBADX;AAEPI,MAAAA,OAAO,EAAE;AAFF,KAAT;;AAIA,QAAI,EAAEC,OAAO,IAAIA,OAAO,CAACC,GAAR,CAAYC,OAAzB,CAAJ,EAAuC;AACrCtH,MAAAA,IAAI,CAACyG,GAAD,CAAJ;AACD;AACF;AACF;;AAEM,SAASe,sBAAT,CAAgCnI,GAAhC,EAAqCU,GAArC,EAA0CC,IAA1C,EAAgD;AACrD,MAAI,CAACX,GAAG,CAACkE,IAAJ,CAASD,QAAd,EAAwB;AACtBvD,IAAAA,GAAG,CAACkH,MAAJ,CAAW,GAAX;AACAlH,IAAAA,GAAG,CAAC0H,GAAJ,CAAQ,kDAAR;AACA;AACD;;AACDzH,EAAAA,IAAI;AACL;;AAEM,SAAS0H,6BAAT,CAAuCC,OAAvC,EAAgD;AACrD,MAAI,CAACA,OAAO,CAACpE,IAAR,CAAaD,QAAlB,EAA4B;AAC1B,UAAMoB,KAAK,GAAG,IAAIE,KAAJ,EAAd;AACAF,IAAAA,KAAK,CAACuC,MAAN,GAAe,GAAf;AACAvC,IAAAA,KAAK,CAACyC,OAAN,GAAgB,sCAAhB;AACA,UAAMzC,KAAN;AACD;;AACD,SAAON,OAAO,CAACC,OAAR,EAAP;AACD;AAED;AACA;AACA;AACA;AACA;AACA;;;AACO,SAASuD,wBAAT,CAAkCvI,GAAlC,EAAuC;AAC5C;AACA,MACE,EACEA,GAAG,CAAC4D,MAAJ,CAAW4E,QAAX,CAAoBC,OAApB,YAAuCC,4BAAvC,IACA1I,GAAG,CAAC4D,MAAJ,CAAW4E,QAAX,CAAoBC,OAApB,YAAuCE,+BAFzC,CADF,EAKE;AACA,WAAO5D,OAAO,CAACC,OAAR,EAAP;AACD,GAT2C,CAU5C;;;AACA,QAAMpB,MAAM,GAAG5D,GAAG,CAAC4D,MAAnB;AACA,QAAMgF,SAAS,GAAG,CAAC,CAAC5I,GAAG,IAAI,EAAR,EAAYkD,OAAZ,IAAuB,EAAxB,EAA4B,oBAA5B,CAAlB;AACA,QAAM;AAAE2F,IAAAA,KAAF;AAASC,IAAAA;AAAT,MAAiBlF,MAAM,CAACmF,kBAA9B;;AACA,MAAI,CAACH,SAAD,IAAc,CAAChF,MAAM,CAACmF,kBAA1B,EAA8C;AAC5C,WAAOhE,OAAO,CAACC,OAAR,EAAP;AACD,GAhB2C,CAiB5C;AACA;;;AACA,QAAMgE,OAAO,GAAGhJ,GAAG,CAACiJ,IAAJ,CAASC,OAAT,CAAiB,SAAjB,EAA4B,EAA5B,CAAhB,CAnB4C,CAoB5C;;AACA,MAAIjD,KAAK,GAAG,KAAZ;;AACA,OAAK,MAAMgD,IAAX,IAAmBJ,KAAnB,EAA0B;AACxB;AACA,UAAMM,KAAK,GAAG,IAAIC,MAAJ,CAAWH,IAAI,CAACI,MAAL,CAAY,CAAZ,MAAmB,GAAnB,GAAyBJ,IAAzB,GAAgC,MAAMA,IAAjD,CAAd;;AACA,QAAID,OAAO,CAAC/C,KAAR,CAAckD,KAAd,CAAJ,EAA0B;AACxBlD,MAAAA,KAAK,GAAG,IAAR;AACA;AACD;AACF;;AACD,MAAI,CAACA,KAAL,EAAY;AACV,WAAOlB,OAAO,CAACC,OAAR,EAAP;AACD,GAhC2C,CAiC5C;;;AACA,QAAMsE,UAAU,GAAG,IAAIC,IAAJ,CAAS,IAAIA,IAAJ,GAAWC,UAAX,CAAsB,IAAID,IAAJ,GAAWE,UAAX,KAA0BX,GAAhD,CAAT,CAAnB;AACA,SAAOY,cACJC,MADI,CACG/F,MADH,EACWM,cAAK0F,MAAL,CAAYhG,MAAZ,CADX,EACgC,cADhC,EACgD;AACnDiG,IAAAA,KAAK,EAAEjB,SAD4C;AAEnDkB,IAAAA,MAAM,EAAExE,cAAMyE,OAAN,CAAcT,UAAd;AAF2C,GADhD,EAKJlE,KALI,CAKEhE,CAAC,IAAI;AACV,QAAIA,CAAC,CAACqG,IAAF,IAAUnC,cAAMC,KAAN,CAAYyE,eAA1B,EAA2C;AACzC,YAAM,IAAI1E,cAAMC,KAAV,CAAgBD,cAAMC,KAAN,CAAY0E,iBAA5B,EAA+C,mBAA/C,CAAN;AACD;;AACD,UAAM7I,CAAN;AACD,GAVI,CAAP;AAWD;;AAED,SAASoB,cAAT,CAAwBxC,GAAxB,EAA6BU,GAA7B,EAAkC;AAChCA,EAAAA,GAAG,CAACkH,MAAJ,CAAW,GAAX;AACAlH,EAAAA,GAAG,CAAC0H,GAAJ,CAAQ,0BAAR;AACD;;AAED,SAAS/G,gBAAT,CAA0BrB,GAA1B,EAA+BU,GAA/B,EAAoC;AAClCA,EAAAA,GAAG,CAACkH,MAAJ,CAAW,GAAX;AACAlH,EAAAA,GAAG,CAACmH,IAAJ,CAAS;AAAEJ,IAAAA,IAAI,EAAEnC,cAAMC,KAAN,CAAY2E,YAApB;AAAkC7E,IAAAA,KAAK,EAAE;AAAzC,GAAT;AACD","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"]}
525
+ //# 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","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","isMaster","ipRangeCheck","masterKeyIps","log","loggerController","defaultLogger","auth","Auth","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","split","jsKeyPrefix","matchKey","str","allowCrossDomain","allowHeaders","join","allowOrigin","method","sendStatus","allowMethodOverride","_method","originalMethod","handleParseErrors","enableExpressErrorHandler","httpStatus","OBJECT_NOT_FOUND","message","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    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  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              throw err;\n            }\n          });\n        }\n      })\n    );\n  } catch (error) {\n    res.status(429);\n    res.json({ code: Parse.Error.CONNECTION_FAILED, error });\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 options.message;\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":";;;;;;;;;;;;;;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAA0C;AAEnC,MAAMA,uBAAuB,GAClC,+OAA+O;AAAC;AAElP,MAAMC,kBAAkB,GAAG,UAAUC,GAAG,EAAE;EACxC,MAAMC,eAAe,GAAGD,GAAG,CAACE,WAAW,CAACC,MAAM,GAAGH,GAAG,CAACI,GAAG,CAACD,MAAM;EAC/D,MAAME,SAAS,GAAGL,GAAG,CAACE,WAAW,CAACI,KAAK,CAAC,CAAC,EAAEL,eAAe,CAAC;EAC3D,OAAOD,GAAG,CAACO,QAAQ,GAAG,KAAK,GAAGP,GAAG,CAACQ,GAAG,CAAC,MAAM,CAAC,GAAGH,SAAS;AAC3D,CAAC;;AAED;AACA;AACA;AACA;AACA;AACA;AACO,SAASI,kBAAkB,CAACT,GAAG,EAAEU,GAAG,EAAEC,IAAI,EAAE;EACjD,IAAIC,KAAK,GAAGb,kBAAkB,CAACC,GAAG,CAAC;EAEnC,IAAIa,OAAO,GAAG,CAAC,CAAC;EAChB,IAAIb,GAAG,CAACQ,GAAG,CAAC,uBAAuB,CAAC,IAAI,IAAI,EAAE;IAC5C,IAAI;MACFK,OAAO,GAAGC,IAAI,CAACC,KAAK,CAACf,GAAG,CAACQ,GAAG,CAAC,uBAAuB,CAAC,CAAC;MACtD,IAAIQ,MAAM,CAACC,SAAS,CAACC,QAAQ,CAACC,IAAI,CAACN,OAAO,CAAC,KAAK,iBAAiB,EAAE;QACjE,MAAM,0BAA0B;MAClC;IACF,CAAC,CAAC,OAAOO,CAAC,EAAE;MACV,OAAOC,gBAAgB,CAACrB,GAAG,EAAEU,GAAG,CAAC;IACnC;EACF;EACA,IAAIY,IAAI,GAAG;IACTC,KAAK,EAAEvB,GAAG,CAACQ,GAAG,CAAC,wBAAwB,CAAC;IACxCgB,YAAY,EAAExB,GAAG,CAACQ,GAAG,CAAC,uBAAuB,CAAC;IAC9CiB,SAAS,EAAEzB,GAAG,CAACQ,GAAG,CAAC,oBAAoB,CAAC;IACxCkB,cAAc,EAAE1B,GAAG,CAACQ,GAAG,CAAC,yBAAyB,CAAC;IAClDmB,SAAS,EAAE3B,GAAG,CAACQ,GAAG,CAAC,oBAAoB,CAAC;IACxCoB,aAAa,EAAE5B,GAAG,CAACQ,GAAG,CAAC,wBAAwB,CAAC;IAChDqB,SAAS,EAAE7B,GAAG,CAACQ,GAAG,CAAC,qBAAqB,CAAC;IACzCsB,UAAU,EAAE9B,GAAG,CAACQ,GAAG,CAAC,sBAAsB,CAAC;IAC3CuB,aAAa,EAAE/B,GAAG,CAACQ,GAAG,CAAC,wBAAwB,CAAC;IAChDK,OAAO,EAAEA;EACX,CAAC;EAED,IAAImB,SAAS,GAAGC,QAAQ,CAACjC,GAAG,CAAC;EAE7B,IAAIgC,SAAS,EAAE;IACb,IAAIE,cAAc,GAAGF,SAAS,CAACT,KAAK;IACpC,IAAIY,cAAQ,CAAC3B,GAAG,CAAC0B,cAAc,CAAC,EAAE;MAChCZ,IAAI,CAACC,KAAK,GAAGW,cAAc;MAC3BZ,IAAI,CAACG,SAAS,GAAGO,SAAS,CAACP,SAAS,IAAIH,IAAI,CAACG,SAAS;MACtDH,IAAI,CAACM,aAAa,GAAGI,SAAS,CAACJ,aAAa,IAAIN,IAAI,CAACM,aAAa;IACpE;EACF;EAEA,IAAI5B,GAAG,CAACoC,IAAI,EAAE;IACZ;IACA;IACA,OAAOpC,GAAG,CAACoC,IAAI,CAACC,OAAO;EACzB;EAEA,IAAIC,WAAW,GAAG,KAAK;EAEvB,IAAI,CAAChB,IAAI,CAACC,KAAK,IAAI,CAACY,cAAQ,CAAC3B,GAAG,CAACc,IAAI,CAACC,KAAK,CAAC,EAAE;IAC5C;IACA,IAAIvB,GAAG,CAACoC,IAAI,YAAYG,MAAM,EAAE;MAC9B;MACA;MACA;MACA;MACA;MACA,IAAI;QACFvC,GAAG,CAACoC,IAAI,GAAGtB,IAAI,CAACC,KAAK,CAACf,GAAG,CAACoC,IAAI,CAAC;MACjC,CAAC,CAAC,OAAOhB,CAAC,EAAE;QACV,OAAOoB,cAAc,CAACxC,GAAG,EAAEU,GAAG,CAAC;MACjC;MACA4B,WAAW,GAAG,IAAI;IACpB;IAEA,IAAItC,GAAG,CAACoC,IAAI,EAAE;MACZ,OAAOpC,GAAG,CAACoC,IAAI,CAACK,iBAAiB;IACnC;IAEA,IACEzC,GAAG,CAACoC,IAAI,IACRpC,GAAG,CAACoC,IAAI,CAACM,cAAc,IACvBP,cAAQ,CAAC3B,GAAG,CAACR,GAAG,CAACoC,IAAI,CAACM,cAAc,CAAC,KACpC,CAACpB,IAAI,CAACG,SAAS,IAAIU,cAAQ,CAAC3B,GAAG,CAACR,GAAG,CAACoC,IAAI,CAACM,cAAc,CAAC,CAACjB,SAAS,KAAKH,IAAI,CAACG,SAAS,CAAC,EACvF;MACAH,IAAI,CAACC,KAAK,GAAGvB,GAAG,CAACoC,IAAI,CAACM,cAAc;MACpCpB,IAAI,CAACM,aAAa,GAAG5B,GAAG,CAACoC,IAAI,CAACO,cAAc,IAAI,EAAE;MAClD,OAAO3C,GAAG,CAACoC,IAAI,CAACM,cAAc;MAC9B,OAAO1C,GAAG,CAACoC,IAAI,CAACO,cAAc;MAC9B;MACA;MACA,IAAI3C,GAAG,CAACoC,IAAI,CAACQ,cAAc,EAAE;QAC3BtB,IAAI,CAACS,aAAa,GAAG/B,GAAG,CAACoC,IAAI,CAACQ,cAAc;QAC5C,OAAO5C,GAAG,CAACoC,IAAI,CAACQ,cAAc;MAChC;MACA,IAAI5C,GAAG,CAACoC,IAAI,CAACS,eAAe,EAAE;QAC5BvB,IAAI,CAACI,cAAc,GAAG1B,GAAG,CAACoC,IAAI,CAACS,eAAe;QAC9C,OAAO7C,GAAG,CAACoC,IAAI,CAACS,eAAe;MACjC;MACA,IAAI7C,GAAG,CAACoC,IAAI,CAACU,aAAa,EAAE;QAC1BxB,IAAI,CAACE,YAAY,GAAGxB,GAAG,CAACoC,IAAI,CAACU,aAAa;QAC1C,OAAO9C,GAAG,CAACoC,IAAI,CAACU,aAAa;MAC/B;MACA,IAAI9C,GAAG,CAACoC,IAAI,CAACW,UAAU,EAAE;QACvBzB,IAAI,CAACG,SAAS,GAAGzB,GAAG,CAACoC,IAAI,CAACW,UAAU;QACpC,OAAO/C,GAAG,CAACoC,IAAI,CAACW,UAAU;MAC5B;MACA,IAAI/C,GAAG,CAACoC,IAAI,CAACY,QAAQ,EAAE;QACrB,IAAIhD,GAAG,CAACoC,IAAI,CAACY,QAAQ,YAAYhC,MAAM,EAAE;UACvCM,IAAI,CAACT,OAAO,GAAGb,GAAG,CAACoC,IAAI,CAACY,QAAQ;QAClC,CAAC,MAAM;UACL,IAAI;YACF1B,IAAI,CAACT,OAAO,GAAGC,IAAI,CAACC,KAAK,CAACf,GAAG,CAACoC,IAAI,CAACY,QAAQ,CAAC;YAC5C,IAAIhC,MAAM,CAACC,SAAS,CAACC,QAAQ,CAACC,IAAI,CAACG,IAAI,CAACT,OAAO,CAAC,KAAK,iBAAiB,EAAE;cACtE,MAAM,0BAA0B;YAClC;UACF,CAAC,CAAC,OAAOO,CAAC,EAAE;YACV,OAAOC,gBAAgB,CAACrB,GAAG,EAAEU,GAAG,CAAC;UACnC;QACF;QACA,OAAOV,GAAG,CAACoC,IAAI,CAACY,QAAQ;MAC1B;MACA,IAAIhD,GAAG,CAACoC,IAAI,CAACa,YAAY,EAAE;QACzBjD,GAAG,CAACkD,OAAO,CAAC,cAAc,CAAC,GAAGlD,GAAG,CAACoC,IAAI,CAACa,YAAY;QACnD,OAAOjD,GAAG,CAACoC,IAAI,CAACa,YAAY;MAC9B;IACF,CAAC,MAAM;MACL,OAAOT,cAAc,CAACxC,GAAG,EAAEU,GAAG,CAAC;IACjC;EACF;EAEA,IAAIY,IAAI,CAACE,YAAY,IAAI,OAAOF,IAAI,CAACE,YAAY,KAAK,QAAQ,EAAE;IAC9DF,IAAI,CAACE,YAAY,GAAGF,IAAI,CAACE,YAAY,CAACN,QAAQ,EAAE;EAClD;EAEA,IAAII,IAAI,CAACS,aAAa,EAAE;IACtBT,IAAI,CAAC6B,SAAS,GAAGC,kBAAS,CAACC,UAAU,CAAC/B,IAAI,CAACS,aAAa,CAAC;EAC3D;EAEA,IAAIO,WAAW,EAAE;IACftC,GAAG,CAACsD,QAAQ,GAAGtD,GAAG,CAACoC,IAAI,CAACkB,QAAQ;IAChC;IACA,IAAIC,MAAM,GAAGvD,GAAG,CAACoC,IAAI,CAACmB,MAAM;IAC5BvD,GAAG,CAACoC,IAAI,GAAGG,MAAM,CAACiB,IAAI,CAACD,MAAM,EAAE,QAAQ,CAAC;EAC1C;EAEA,MAAME,QAAQ,GAAGC,WAAW,CAAC1D,GAAG,CAAC;EACjC,MAAM2D,MAAM,GAAGC,eAAM,CAACpD,GAAG,CAACc,IAAI,CAACC,KAAK,EAAEX,KAAK,CAAC;EAC5C,IAAI+C,MAAM,CAACE,KAAK,IAAIF,MAAM,CAACE,KAAK,KAAK,IAAI,EAAE;IACzCnD,GAAG,CAACoD,MAAM,CAAC,GAAG,CAAC;IACfpD,GAAG,CAACqD,IAAI,CAAC;MACPC,IAAI,EAAEC,aAAK,CAACC,KAAK,CAACC,qBAAqB;MACvCC,KAAK,EAAG,yBAAwBT,MAAM,CAACE,KAAM;IAC/C,CAAC,CAAC;IACF;EACF;EAEAvC,IAAI,CAAC+C,GAAG,GAAGlC,cAAQ,CAAC3B,GAAG,CAACc,IAAI,CAACC,KAAK,CAAC;EACnCvB,GAAG,CAAC2D,MAAM,GAAGA,MAAM;EACnB3D,GAAG,CAAC2D,MAAM,CAACT,OAAO,GAAGlD,GAAG,CAACkD,OAAO,IAAI,CAAC,CAAC;EACtClD,GAAG,CAAC2D,MAAM,CAACW,EAAE,GAAGb,QAAQ;EACxBzD,GAAG,CAACsB,IAAI,GAAGA,IAAI;EAEf,IAAIiD,QAAQ,GAAGjD,IAAI,CAACG,SAAS,KAAKzB,GAAG,CAAC2D,MAAM,CAAClC,SAAS;EACtD,IAAI8C,QAAQ,IAAI,CAAC,IAAAC,qBAAY,EAACf,QAAQ,EAAEzD,GAAG,CAAC2D,MAAM,CAACc,YAAY,IAAI,EAAE,CAAC,EAAE;IAAA;IACtE,MAAMC,GAAG,GAAG,gBAAA1E,GAAG,CAAC2D,MAAM,gDAAV,YAAYgB,gBAAgB,KAAIC,eAAa;IACzDF,GAAG,CAACN,KAAK,CACN,gEAA+DX,QAAS,qDAAoD,CAC9H;IACDc,QAAQ,GAAG,KAAK;EAClB;EAEA,IAAIA,QAAQ,EAAE;IACZvE,GAAG,CAAC6E,IAAI,GAAG,IAAIA,aAAI,CAACC,IAAI,CAAC;MACvBnB,MAAM,EAAE3D,GAAG,CAAC2D,MAAM;MAClBjC,cAAc,EAAEJ,IAAI,CAACI,cAAc;MACnC6C,QAAQ,EAAE;IACZ,CAAC,CAAC;IACF,OAAOQ,eAAe,CAAC/E,GAAG,EAAEU,GAAG,EAAEC,IAAI,CAAC;EACxC;EAEA,IAAIqE,gBAAgB,GAAG1D,IAAI,CAACG,SAAS,KAAKzB,GAAG,CAAC2D,MAAM,CAACsB,iBAAiB;EACtE,IACE,OAAOjF,GAAG,CAAC2D,MAAM,CAACsB,iBAAiB,IAAI,WAAW,IAClDjF,GAAG,CAAC2D,MAAM,CAACsB,iBAAiB,IAC5BD,gBAAgB,EAChB;IACAhF,GAAG,CAAC6E,IAAI,GAAG,IAAIA,aAAI,CAACC,IAAI,CAAC;MACvBnB,MAAM,EAAE3D,GAAG,CAAC2D,MAAM;MAClBjC,cAAc,EAAEJ,IAAI,CAACI,cAAc;MACnC6C,QAAQ,EAAE,IAAI;MACdW,UAAU,EAAE;IACd,CAAC,CAAC;IACF,OAAOH,eAAe,CAAC/E,GAAG,EAAEU,GAAG,EAAEC,IAAI,CAAC;EACxC;;EAEA;EACA;EACA,MAAMwE,IAAI,GAAG,CAAC,WAAW,EAAE,eAAe,EAAE,WAAW,EAAE,YAAY,CAAC;EACtE,MAAMC,gBAAgB,GAAGD,IAAI,CAACE,IAAI,CAAC,UAAUC,GAAG,EAAE;IAChD,OAAOtF,GAAG,CAAC2D,MAAM,CAAC2B,GAAG,CAAC,KAAKC,SAAS;EACtC,CAAC,CAAC;EACF,MAAMC,aAAa,GAAGL,IAAI,CAACE,IAAI,CAAC,UAAUC,GAAG,EAAE;IAC7C,OAAOtF,GAAG,CAAC2D,MAAM,CAAC2B,GAAG,CAAC,KAAKC,SAAS,IAAIjE,IAAI,CAACgE,GAAG,CAAC,KAAKtF,GAAG,CAAC2D,MAAM,CAAC2B,GAAG,CAAC;EACvE,CAAC,CAAC;EAEF,IAAIF,gBAAgB,IAAI,CAACI,aAAa,EAAE;IACtC,OAAOhD,cAAc,CAACxC,GAAG,EAAEU,GAAG,CAAC;EACjC;EAEA,IAAIV,GAAG,CAACI,GAAG,IAAI,QAAQ,EAAE;IACvB,OAAOkB,IAAI,CAACE,YAAY;EAC1B;EAEA,IAAIxB,GAAG,CAACyF,WAAW,EAAE;IACnBzF,GAAG,CAAC6E,IAAI,GAAG,IAAIA,aAAI,CAACC,IAAI,CAAC;MACvBnB,MAAM,EAAE3D,GAAG,CAAC2D,MAAM;MAClBjC,cAAc,EAAEJ,IAAI,CAACI,cAAc;MACnC6C,QAAQ,EAAE,KAAK;MACfmB,IAAI,EAAE1F,GAAG,CAACyF;IACZ,CAAC,CAAC;IACF,OAAOV,eAAe,CAAC/E,GAAG,EAAEU,GAAG,EAAEC,IAAI,CAAC;EACxC;EAEA,IAAI,CAACW,IAAI,CAACE,YAAY,EAAE;IACtBxB,GAAG,CAAC6E,IAAI,GAAG,IAAIA,aAAI,CAACC,IAAI,CAAC;MACvBnB,MAAM,EAAE3D,GAAG,CAAC2D,MAAM;MAClBjC,cAAc,EAAEJ,IAAI,CAACI,cAAc;MACnC6C,QAAQ,EAAE;IACZ,CAAC,CAAC;EACJ;EACAQ,eAAe,CAAC/E,GAAG,EAAEU,GAAG,EAAEC,IAAI,CAAC;AACjC;AAEA,MAAMoE,eAAe,GAAG,OAAO/E,GAAG,EAAEU,GAAG,EAAEC,IAAI,KAAK;EAChD,MAAMgF,UAAU,GAAG3F,GAAG,CAAC2D,MAAM,CAACgC,UAAU,IAAI,EAAE;EAC9C,IAAI;IACF,MAAMC,OAAO,CAACC,GAAG,CACfF,UAAU,CAACG,GAAG,CAAC,MAAMC,KAAK,IAAI;MAC5B,MAAMC,OAAO,GAAG,IAAIC,MAAM,CAACF,KAAK,CAACG,IAAI,CAAC;MACtC,IAAIF,OAAO,CAACG,IAAI,CAACnG,GAAG,CAACI,GAAG,CAAC,EAAE;QACzB,MAAM2F,KAAK,CAACK,OAAO,CAACpG,GAAG,EAAEU,GAAG,EAAE2F,GAAG,IAAI;UACnC,IAAIA,GAAG,EAAE;YACP,MAAMA,GAAG;UACX;QACF,CAAC,CAAC;MACJ;IACF,CAAC,CAAC,CACH;EACH,CAAC,CAAC,OAAOjC,KAAK,EAAE;IACd1D,GAAG,CAACoD,MAAM,CAAC,GAAG,CAAC;IACfpD,GAAG,CAACqD,IAAI,CAAC;MAAEC,IAAI,EAAEC,aAAK,CAACC,KAAK,CAACoC,iBAAiB;MAAElC;IAAM,CAAC,CAAC;IACxD;EACF;EACAzD,IAAI,EAAE;AACR,CAAC;AAEM,MAAM4F,kBAAkB,GAAG,OAAOvG,GAAG,EAAEU,GAAG,EAAEC,IAAI,KAAK;EAC1D,IAAI;IACF,MAAMW,IAAI,GAAGtB,GAAG,CAACsB,IAAI;IACrB,IAAItB,GAAG,CAAC6E,IAAI,EAAE;MACZlE,IAAI,EAAE;MACN;IACF;IACA,IAAI6F,WAAW,GAAG,IAAI;IACtB,IACElF,IAAI,CAACE,YAAY,IACjBxB,GAAG,CAACI,GAAG,KAAK,4BAA4B,IACxCkB,IAAI,CAACE,YAAY,CAACiF,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,EACpC;MACAD,WAAW,GAAG,MAAM3B,aAAI,CAAC6B,4BAA4B,CAAC;QACpD/C,MAAM,EAAE3D,GAAG,CAAC2D,MAAM;QAClBjC,cAAc,EAAEJ,IAAI,CAACI,cAAc;QACnCF,YAAY,EAAEF,IAAI,CAACE;MACrB,CAAC,CAAC;IACJ,CAAC,MAAM;MACLgF,WAAW,GAAG,MAAM3B,aAAI,CAAC8B,sBAAsB,CAAC;QAC9ChD,MAAM,EAAE3D,GAAG,CAAC2D,MAAM;QAClBjC,cAAc,EAAEJ,IAAI,CAACI,cAAc;QACnCF,YAAY,EAAEF,IAAI,CAACE;MACrB,CAAC,CAAC;IACJ;IACAxB,GAAG,CAAC6E,IAAI,GAAG2B,WAAW;IACtB7F,IAAI,EAAE;EACR,CAAC,CAAC,OAAOyD,KAAK,EAAE;IACd,IAAIA,KAAK,YAAYH,aAAK,CAACC,KAAK,EAAE;MAChCvD,IAAI,CAACyD,KAAK,CAAC;MACX;IACF;IACA;IACApE,GAAG,CAAC2D,MAAM,CAACgB,gBAAgB,CAACP,KAAK,CAAC,qCAAqC,EAAEA,KAAK,CAAC;IAC/E,MAAM,IAAIH,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAAC0C,aAAa,EAAExC,KAAK,CAAC;EACzD;AACF,CAAC;AAAC;AAEF,SAASV,WAAW,CAAC1D,GAAG,EAAE;EACxB,OAAOA,GAAG,CAACsE,EAAE;AACf;AAEA,SAASrC,QAAQ,CAACjC,GAAG,EAAE;EACrB,IAAI,CAAC,CAACA,GAAG,CAACA,GAAG,IAAIA,GAAG,EAAEkD,OAAO,CAAC2D,aAAa,EAAE;EAE7C,IAAIC,MAAM,GAAG,CAAC9G,GAAG,CAACA,GAAG,IAAIA,GAAG,EAAEkD,OAAO,CAAC2D,aAAa;EACnD,IAAItF,KAAK,EAAEE,SAAS,EAAEG,aAAa;;EAEnC;EACA,IAAImF,UAAU,GAAG,QAAQ;EAEzB,IAAIC,KAAK,GAAGF,MAAM,CAACG,WAAW,EAAE,CAACR,OAAO,CAACM,UAAU,CAAC;EAEpD,IAAIC,KAAK,IAAI,CAAC,EAAE;IACd,IAAIE,WAAW,GAAGJ,MAAM,CAACK,SAAS,CAACJ,UAAU,CAAC5G,MAAM,EAAE2G,MAAM,CAAC3G,MAAM,CAAC;IACpE,IAAIiH,WAAW,GAAGC,YAAY,CAACH,WAAW,CAAC,CAACI,KAAK,CAAC,GAAG,CAAC;IAEtD,IAAIF,WAAW,CAACjH,MAAM,IAAI,CAAC,EAAE;MAC3BoB,KAAK,GAAG6F,WAAW,CAAC,CAAC,CAAC;MACtB,IAAI9B,GAAG,GAAG8B,WAAW,CAAC,CAAC,CAAC;MAExB,IAAIG,WAAW,GAAG,iBAAiB;MAEnC,IAAIC,QAAQ,GAAGlC,GAAG,CAACmB,OAAO,CAACc,WAAW,CAAC;MACvC,IAAIC,QAAQ,IAAI,CAAC,EAAE;QACjB5F,aAAa,GAAG0D,GAAG,CAAC6B,SAAS,CAACI,WAAW,CAACpH,MAAM,EAAEmF,GAAG,CAACnF,MAAM,CAAC;MAC/D,CAAC,MAAM;QACLsB,SAAS,GAAG6D,GAAG;MACjB;IACF;EACF;EAEA,OAAO;IAAE/D,KAAK,EAAEA,KAAK;IAAEE,SAAS,EAAEA,SAAS;IAAEG,aAAa,EAAEA;EAAc,CAAC;AAC7E;AAEA,SAASyF,YAAY,CAACI,GAAG,EAAE;EACzB,OAAOlF,MAAM,CAACiB,IAAI,CAACiE,GAAG,EAAE,QAAQ,CAAC,CAACvG,QAAQ,EAAE;AAC9C;AAEO,SAASwG,gBAAgB,CAACnG,KAAK,EAAE;EACtC,OAAO,CAACvB,GAAG,EAAEU,GAAG,EAAEC,IAAI,KAAK;IACzB,MAAMgD,MAAM,GAAGC,eAAM,CAACpD,GAAG,CAACe,KAAK,EAAExB,kBAAkB,CAACC,GAAG,CAAC,CAAC;IACzD,IAAI2H,YAAY,GAAG7H,uBAAuB;IAC1C,IAAI6D,MAAM,IAAIA,MAAM,CAACgE,YAAY,EAAE;MACjCA,YAAY,IAAK,KAAIhE,MAAM,CAACgE,YAAY,CAACC,IAAI,CAAC,IAAI,CAAE,EAAC;IACvD;IACA,MAAMC,WAAW,GAAIlE,MAAM,IAAIA,MAAM,CAACkE,WAAW,IAAK,GAAG;IACzDnH,GAAG,CAACoG,MAAM,CAAC,6BAA6B,EAAEe,WAAW,CAAC;IACtDnH,GAAG,CAACoG,MAAM,CAAC,8BAA8B,EAAE,6BAA6B,CAAC;IACzEpG,GAAG,CAACoG,MAAM,CAAC,8BAA8B,EAAEa,YAAY,CAAC;IACxDjH,GAAG,CAACoG,MAAM,CAAC,+BAA+B,EAAE,+CAA+C,CAAC;IAC5F;IACA,IAAI,SAAS,IAAI9G,GAAG,CAAC8H,MAAM,EAAE;MAC3BpH,GAAG,CAACqH,UAAU,CAAC,GAAG,CAAC;IACrB,CAAC,MAAM;MACLpH,IAAI,EAAE;IACR;EACF,CAAC;AACH;AAEO,SAASqH,mBAAmB,CAAChI,GAAG,EAAEU,GAAG,EAAEC,IAAI,EAAE;EAClD,IAAIX,GAAG,CAAC8H,MAAM,KAAK,MAAM,IAAI9H,GAAG,CAACoC,IAAI,CAAC6F,OAAO,EAAE;IAC7CjI,GAAG,CAACkI,cAAc,GAAGlI,GAAG,CAAC8H,MAAM;IAC/B9H,GAAG,CAAC8H,MAAM,GAAG9H,GAAG,CAACoC,IAAI,CAAC6F,OAAO;IAC7B,OAAOjI,GAAG,CAACoC,IAAI,CAAC6F,OAAO;EACzB;EACAtH,IAAI,EAAE;AACR;AAEO,SAASwH,iBAAiB,CAAC9B,GAAG,EAAErG,GAAG,EAAEU,GAAG,EAAEC,IAAI,EAAE;EACrD,MAAM+D,GAAG,GAAI1E,GAAG,CAAC2D,MAAM,IAAI3D,GAAG,CAAC2D,MAAM,CAACgB,gBAAgB,IAAKC,eAAa;EACxE,IAAIyB,GAAG,YAAYpC,aAAK,CAACC,KAAK,EAAE;IAC9B,IAAIlE,GAAG,CAAC2D,MAAM,IAAI3D,GAAG,CAAC2D,MAAM,CAACyE,yBAAyB,EAAE;MACtD,OAAOzH,IAAI,CAAC0F,GAAG,CAAC;IAClB;IACA,IAAIgC,UAAU;IACd;IACA,QAAQhC,GAAG,CAACrC,IAAI;MACd,KAAKC,aAAK,CAACC,KAAK,CAACC,qBAAqB;QACpCkE,UAAU,GAAG,GAAG;QAChB;MACF,KAAKpE,aAAK,CAACC,KAAK,CAACoE,gBAAgB;QAC/BD,UAAU,GAAG,GAAG;QAChB;MACF;QACEA,UAAU,GAAG,GAAG;IAAC;IAErB3H,GAAG,CAACoD,MAAM,CAACuE,UAAU,CAAC;IACtB3H,GAAG,CAACqD,IAAI,CAAC;MAAEC,IAAI,EAAEqC,GAAG,CAACrC,IAAI;MAAEI,KAAK,EAAEiC,GAAG,CAACkC;IAAQ,CAAC,CAAC;IAChD7D,GAAG,CAACN,KAAK,CAAC,eAAe,EAAEiC,GAAG,CAAC;EACjC,CAAC,MAAM,IAAIA,GAAG,CAACvC,MAAM,IAAIuC,GAAG,CAACkC,OAAO,EAAE;IACpC7H,GAAG,CAACoD,MAAM,CAACuC,GAAG,CAACvC,MAAM,CAAC;IACtBpD,GAAG,CAACqD,IAAI,CAAC;MAAEK,KAAK,EAAEiC,GAAG,CAACkC;IAAQ,CAAC,CAAC;IAChC,IAAI,EAAEC,OAAO,IAAIA,OAAO,CAACC,GAAG,CAACC,OAAO,CAAC,EAAE;MACrC/H,IAAI,CAAC0F,GAAG,CAAC;IACX;EACF,CAAC,MAAM;IACL3B,GAAG,CAACN,KAAK,CAAC,iCAAiC,EAAEiC,GAAG,EAAEA,GAAG,CAACsC,KAAK,CAAC;IAC5DjI,GAAG,CAACoD,MAAM,CAAC,GAAG,CAAC;IACfpD,GAAG,CAACqD,IAAI,CAAC;MACPC,IAAI,EAAEC,aAAK,CAACC,KAAK,CAACC,qBAAqB;MACvCoE,OAAO,EAAE;IACX,CAAC,CAAC;IACF,IAAI,EAAEC,OAAO,IAAIA,OAAO,CAACC,GAAG,CAACC,OAAO,CAAC,EAAE;MACrC/H,IAAI,CAAC0F,GAAG,CAAC;IACX;EACF;AACF;AAEO,SAASuC,sBAAsB,CAAC5I,GAAG,EAAEU,GAAG,EAAEC,IAAI,EAAE;EACrD,IAAI,CAACX,GAAG,CAAC6E,IAAI,CAACN,QAAQ,EAAE;IACtB7D,GAAG,CAACoD,MAAM,CAAC,GAAG,CAAC;IACfpD,GAAG,CAACmI,GAAG,CAAC,kDAAkD,CAAC;IAC3D;EACF;EACAlI,IAAI,EAAE;AACR;AAEO,SAASmI,6BAA6B,CAACC,OAAO,EAAE;EACrD,IAAI,CAACA,OAAO,CAAClE,IAAI,CAACN,QAAQ,EAAE;IAC1B,MAAMH,KAAK,GAAG,IAAIF,KAAK,EAAE;IACzBE,KAAK,CAACN,MAAM,GAAG,GAAG;IAClBM,KAAK,CAACmE,OAAO,GAAG,sCAAsC;IACtD,MAAMnE,KAAK;EACb;EACA,OAAOwB,OAAO,CAACoD,OAAO,EAAE;AAC1B;AAEO,MAAMC,YAAY,GAAG,CAACC,KAAK,EAAEvF,MAAM,KAAK;EAC7C,IAAI,OAAOA,MAAM,KAAK,QAAQ,EAAE;IAC9BA,MAAM,GAAGC,eAAM,CAACpD,GAAG,CAACmD,MAAM,CAAC;EAC7B;EACA,KAAK,MAAM2B,GAAG,IAAI4D,KAAK,EAAE;IACvB,IAAI,CAACC,6BAAgB,CAAC7D,GAAG,CAAC,EAAE;MAC1B,MAAO,8BAA6BA,GAAI,GAAE;IAC5C;EACF;EACA,IAAI,CAAC3B,MAAM,CAACgC,UAAU,EAAE;IACtBhC,MAAM,CAACgC,UAAU,GAAG,EAAE;EACxB;EACAhC,MAAM,CAACgC,UAAU,CAACyD,IAAI,CAAC;IACrBlD,IAAI,EAAE,IAAAmD,qBAAY,EAACH,KAAK,CAACI,WAAW,CAAC;IACrClD,OAAO,EAAE,IAAAmD,yBAAS,EAAC;MACjBC,QAAQ,EAAEN,KAAK,CAACO,iBAAiB;MACjCC,GAAG,EAAER,KAAK,CAACS,YAAY;MACvBpB,OAAO,EAAEW,KAAK,CAACU,oBAAoB,IAAIT,6BAAgB,CAACS,oBAAoB,CAACC,OAAO;MACpFzD,OAAO,EAAE,CAAC2C,OAAO,EAAEe,QAAQ,EAAEnJ,IAAI,EAAEoJ,OAAO,KAAK;QAC7C,MAAMA,OAAO,CAACxB,OAAO;MACvB,CAAC;MACDyB,IAAI,EAAEjB,OAAO,IAAI;QACf,IAAIA,OAAO,CAACzE,EAAE,KAAK,WAAW,IAAI,CAAC4E,KAAK,CAACe,uBAAuB,EAAE;UAChE,OAAO,IAAI;QACb;QACA,IAAIf,KAAK,CAACgB,gBAAgB,EAAE;UAC1B,OAAO,KAAK;QACd;QACA,IAAIhB,KAAK,CAACiB,cAAc,EAAE;UACxB,IAAIC,KAAK,CAACC,OAAO,CAACnB,KAAK,CAACiB,cAAc,CAAC,EAAE;YACvC,IAAI,CAACjB,KAAK,CAACiB,cAAc,CAACG,QAAQ,CAACvB,OAAO,CAACjB,MAAM,CAAC,EAAE;cAClD,OAAO,IAAI;YACb;UACF,CAAC,MAAM;YACL,MAAMyC,MAAM,GAAG,IAAItE,MAAM,CAACiD,KAAK,CAACiB,cAAc,CAAC;YAC/C,IAAI,CAACI,MAAM,CAACpE,IAAI,CAAC4C,OAAO,CAACjB,MAAM,CAAC,EAAE;cAChC,OAAO,IAAI;YACb;UACF;QACF;QACA,OAAOiB,OAAO,CAAClE,IAAI,CAACN,QAAQ;MAC9B,CAAC;MACDiG,YAAY,EAAEzB,OAAO,IAAI;QACvB,OAAOA,OAAO,CAACpF,MAAM,CAACW,EAAE;MAC1B;IACF,CAAC;EACH,CAAC,CAAC;EACFV,eAAM,CAAC6G,GAAG,CAAC9G,MAAM,CAAC;AACpB,CAAC;;AAED;AACA;AACA;AACA;AACA;AACA;AALA;AAMO,SAAS+G,wBAAwB,CAAC1K,GAAG,EAAE;EAC5C;EACA,IACE,EACEA,GAAG,CAAC2D,MAAM,CAACgH,QAAQ,CAACC,OAAO,YAAYC,4BAAmB,IAC1D7K,GAAG,CAAC2D,MAAM,CAACgH,QAAQ,CAACC,OAAO,YAAYE,+BAAsB,CAC9D,EACD;IACA,OAAOlF,OAAO,CAACoD,OAAO,EAAE;EAC1B;EACA;EACA,MAAMrF,MAAM,GAAG3D,GAAG,CAAC2D,MAAM;EACzB,MAAMoH,SAAS,GAAG,CAAC,CAAC/K,GAAG,IAAI,CAAC,CAAC,EAAEkD,OAAO,IAAI,CAAC,CAAC,EAAE,oBAAoB,CAAC;EACnE,MAAM;IAAE8H,KAAK;IAAEC;EAAI,CAAC,GAAGtH,MAAM,CAACuH,kBAAkB;EAChD,IAAI,CAACH,SAAS,IAAI,CAACpH,MAAM,CAACuH,kBAAkB,EAAE;IAC5C,OAAOtF,OAAO,CAACoD,OAAO,EAAE;EAC1B;EACA;EACA;EACA,MAAMmC,OAAO,GAAGnL,GAAG,CAACkG,IAAI,CAACkF,OAAO,CAAC,SAAS,EAAE,EAAE,CAAC;EAC/C;EACA,IAAIpE,KAAK,GAAG,KAAK;EACjB,KAAK,MAAMd,IAAI,IAAI8E,KAAK,EAAE;IACxB;IACA,MAAMK,KAAK,GAAG,IAAIpF,MAAM,CAACC,IAAI,CAACoF,MAAM,CAAC,CAAC,CAAC,KAAK,GAAG,GAAGpF,IAAI,GAAG,GAAG,GAAGA,IAAI,CAAC;IACpE,IAAIiF,OAAO,CAACnE,KAAK,CAACqE,KAAK,CAAC,EAAE;MACxBrE,KAAK,GAAG,IAAI;MACZ;IACF;EACF;EACA,IAAI,CAACA,KAAK,EAAE;IACV,OAAOpB,OAAO,CAACoD,OAAO,EAAE;EAC1B;EACA;EACA,MAAMuC,UAAU,GAAG,IAAIC,IAAI,CAAC,IAAIA,IAAI,EAAE,CAACC,UAAU,CAAC,IAAID,IAAI,EAAE,CAACE,UAAU,EAAE,GAAGT,GAAG,CAAC,CAAC;EACjF,OAAOU,aAAI,CACRC,MAAM,CAACjI,MAAM,EAAEkB,aAAI,CAACgH,MAAM,CAAClI,MAAM,CAAC,EAAE,cAAc,EAAE;IACnDmI,KAAK,EAAEf,SAAS;IAChBgB,MAAM,EAAE9H,aAAK,CAAC+H,OAAO,CAACT,UAAU;EAClC,CAAC,CAAC,CACDU,KAAK,CAAC7K,CAAC,IAAI;IACV,IAAIA,CAAC,CAAC4C,IAAI,IAAIC,aAAK,CAACC,KAAK,CAACgI,eAAe,EAAE;MACzC,MAAM,IAAIjI,aAAK,CAACC,KAAK,CAACD,aAAK,CAACC,KAAK,CAACiI,iBAAiB,EAAE,mBAAmB,CAAC;IAC3E;IACA,MAAM/K,CAAC;EACT,CAAC,CAAC;AACN;AAEA,SAASoB,cAAc,CAACxC,GAAG,EAAEU,GAAG,EAAE;EAChCA,GAAG,CAACoD,MAAM,CAAC,GAAG,CAAC;EACfpD,GAAG,CAACmI,GAAG,CAAC,0BAA0B,CAAC;AACrC;AAEA,SAASxH,gBAAgB,CAACrB,GAAG,EAAEU,GAAG,EAAE;EAClCA,GAAG,CAACoD,MAAM,CAAC,GAAG,CAAC;EACfpD,GAAG,CAACqD,IAAI,CAAC;IAAEC,IAAI,EAAEC,aAAK,CAACC,KAAK,CAACkI,YAAY;IAAEhI,KAAK,EAAE;EAA8B,CAAC,CAAC;AACpF"}