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
@@ -4,27 +4,16 @@ Object.defineProperty(exports, "__esModule", {
4
4
  value: true
5
5
  });
6
6
  exports.default = exports.PagesRouter = void 0;
7
-
8
7
  var _PromiseRouter = _interopRequireDefault(require("../PromiseRouter"));
9
-
10
8
  var _Config = _interopRequireDefault(require("../Config"));
11
-
12
9
  var _express = _interopRequireDefault(require("express"));
13
-
14
10
  var _path = _interopRequireDefault(require("path"));
15
-
16
11
  var _fs = require("fs");
17
-
18
12
  var _node = require("parse/node");
19
-
20
13
  var _Utils = _interopRequireDefault(require("../Utils"));
21
-
22
14
  var _mustache = _interopRequireDefault(require("mustache"));
23
-
24
15
  var _Page = _interopRequireDefault(require("../Page"));
25
-
26
16
  function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
27
-
28
17
  // All pages with custom page key for reference and file name
29
18
  const pages = Object.freeze({
30
19
  passwordReset: new _Page.default({
@@ -59,8 +48,9 @@ const pages = Object.freeze({
59
48
  id: 'emailVerificationLinkExpired',
60
49
  defaultFile: 'email_verification_link_expired.html'
61
50
  })
62
- }); // All page parameters for reference to be used as template placeholders or query params
51
+ });
63
52
 
53
+ // All page parameters for reference to be used as template placeholders or query params
64
54
  const pageParams = Object.freeze({
65
55
  appName: 'appName',
66
56
  appId: 'appId',
@@ -69,23 +59,25 @@ const pageParams = Object.freeze({
69
59
  error: 'error',
70
60
  locale: 'locale',
71
61
  publicServerUrl: 'publicServerUrl'
72
- }); // The header prefix to add page params as response headers
62
+ });
73
63
 
74
- const pageParamHeaderPrefix = 'x-parse-page-param-'; // The errors being thrown
64
+ // The header prefix to add page params as response headers
65
+ const pageParamHeaderPrefix = 'x-parse-page-param-';
75
66
 
67
+ // The errors being thrown
76
68
  const errors = Object.freeze({
77
69
  jsonFailedFileLoading: 'failed to load JSON file',
78
70
  fileOutsideAllowedScope: 'not allowed to read file outside of pages directory'
79
71
  });
80
-
81
72
  class PagesRouter extends _PromiseRouter.default {
82
73
  /**
83
74
  * Constructs a PagesRouter.
84
75
  * @param {Object} pages The pages options from the Parse Server configuration.
85
76
  */
86
77
  constructor(pages = {}) {
87
- super(); // Set instance properties
78
+ super();
88
79
 
80
+ // Set instance properties
89
81
  this.pagesConfig = pages;
90
82
  this.pagesEndpoint = pages.pagesEndpoint ? pages.pagesEndpoint : 'apps';
91
83
  this.pagesPath = pages.pagesPath ? _path.default.resolve('./', pages.pagesPath) : _path.default.resolve(__dirname, '../../public');
@@ -94,7 +86,6 @@ class PagesRouter extends _PromiseRouter.default {
94
86
  this.mountCustomRoutes();
95
87
  this.mountStaticRoute();
96
88
  }
97
-
98
89
  verifyEmail(req) {
99
90
  const config = req.config;
100
91
  const {
@@ -102,15 +93,12 @@ class PagesRouter extends _PromiseRouter.default {
102
93
  token: rawToken
103
94
  } = req.query;
104
95
  const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;
105
-
106
96
  if (!config) {
107
97
  this.invalidRequest();
108
98
  }
109
-
110
99
  if (!token || !username) {
111
100
  return this.goToPage(req, pages.emailVerificationLinkInvalid);
112
101
  }
113
-
114
102
  const userController = config.userController;
115
103
  return userController.verifyEmail(username, token).then(() => {
116
104
  const params = {
@@ -124,19 +112,15 @@ class PagesRouter extends _PromiseRouter.default {
124
112
  return this.goToPage(req, pages.emailVerificationLinkExpired, params);
125
113
  });
126
114
  }
127
-
128
115
  resendVerificationEmail(req) {
129
116
  const config = req.config;
130
117
  const username = req.body.username;
131
-
132
118
  if (!config) {
133
119
  this.invalidRequest();
134
120
  }
135
-
136
121
  if (!username) {
137
122
  return this.goToPage(req, pages.emailVerificationLinkInvalid);
138
123
  }
139
-
140
124
  const userController = config.userController;
141
125
  return userController.resendVerificationEmail(username).then(() => {
142
126
  return this.goToPage(req, pages.emailVerificationSendSuccess);
@@ -144,7 +128,6 @@ class PagesRouter extends _PromiseRouter.default {
144
128
  return this.goToPage(req, pages.emailVerificationSendFail);
145
129
  });
146
130
  }
147
-
148
131
  passwordReset(req) {
149
132
  const config = req.config;
150
133
  const params = {
@@ -156,24 +139,19 @@ class PagesRouter extends _PromiseRouter.default {
156
139
  };
157
140
  return this.goToPage(req, pages.passwordReset, params);
158
141
  }
159
-
160
142
  requestResetPassword(req) {
161
143
  const config = req.config;
162
-
163
144
  if (!config) {
164
145
  this.invalidRequest();
165
146
  }
166
-
167
147
  const {
168
148
  username,
169
149
  token: rawToken
170
150
  } = req.query;
171
151
  const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;
172
-
173
152
  if (!username || !token) {
174
153
  return this.goToPage(req, pages.passwordResetLinkInvalid);
175
154
  }
176
-
177
155
  return config.userController.checkResetTokenValidity(username, token).then(() => {
178
156
  const params = {
179
157
  [pageParams.token]: token,
@@ -189,37 +167,29 @@ class PagesRouter extends _PromiseRouter.default {
189
167
  return this.goToPage(req, pages.passwordResetLinkInvalid, params);
190
168
  });
191
169
  }
192
-
193
170
  resetPassword(req) {
194
171
  const config = req.config;
195
-
196
172
  if (!config) {
197
173
  this.invalidRequest();
198
174
  }
199
-
200
175
  const {
201
176
  username,
202
177
  new_password,
203
178
  token: rawToken
204
179
  } = req.body;
205
180
  const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;
206
-
207
181
  if ((!username || !token || !new_password) && req.xhr === false) {
208
182
  return this.goToPage(req, pages.passwordResetLinkInvalid);
209
183
  }
210
-
211
184
  if (!username) {
212
185
  throw new _node.Parse.Error(_node.Parse.Error.USERNAME_MISSING, 'Missing username');
213
186
  }
214
-
215
187
  if (!token) {
216
188
  throw new _node.Parse.Error(_node.Parse.Error.OTHER_CAUSE, 'Missing token');
217
189
  }
218
-
219
190
  if (!new_password) {
220
191
  throw new _node.Parse.Error(_node.Parse.Error.PASSWORD_MISSING, 'Missing password');
221
192
  }
222
-
223
193
  return config.userController.updatePassword(username, token, new_password).then(() => {
224
194
  return Promise.resolve({
225
195
  success: true
@@ -237,12 +207,10 @@ class PagesRouter extends _PromiseRouter.default {
237
207
  response: 'Password successfully reset'
238
208
  });
239
209
  }
240
-
241
210
  if (result.err) {
242
211
  throw new _node.Parse.Error(_node.Parse.Error.OTHER_CAUSE, `${result.err}`);
243
212
  }
244
213
  }
245
-
246
214
  const query = result.success ? {
247
215
  [pageParams.username]: username
248
216
  } : {
@@ -256,6 +224,7 @@ class PagesRouter extends _PromiseRouter.default {
256
224
  return this.goToPage(req, page, query, false);
257
225
  });
258
226
  }
227
+
259
228
  /**
260
229
  * Returns page content if the page is a local file or returns a
261
230
  * redirect to a custom page.
@@ -271,44 +240,43 @@ class PagesRouter extends _PromiseRouter.default {
271
240
  * - POST request -> redirect response (PRG pattern)
272
241
  * @returns {Promise<Object>} The PromiseRouter response.
273
242
  */
274
-
275
-
276
243
  goToPage(req, page, params = {}, responseType) {
277
- const config = req.config; // Determine redirect either by force, response setting or request method
244
+ const config = req.config;
278
245
 
279
- const redirect = config.pages.forceRedirect ? true : responseType !== undefined ? responseType : req.method == 'POST'; // Include default parameters
246
+ // Determine redirect either by force, response setting or request method
247
+ const redirect = config.pages.forceRedirect ? true : responseType !== undefined ? responseType : req.method == 'POST';
280
248
 
249
+ // Include default parameters
281
250
  const defaultParams = this.getDefaultParams(config);
282
-
283
251
  if (Object.values(defaultParams).includes(undefined)) {
284
252
  return this.notFound();
285
253
  }
254
+ params = Object.assign(params, defaultParams);
286
255
 
287
- params = Object.assign(params, defaultParams); // Add locale to params to ensure it is passed on with every request;
256
+ // Add locale to params to ensure it is passed on with every request;
288
257
  // that means, once a locale is set, it is passed on to any follow-up page,
289
258
  // e.g. request_password_reset -> password_reset -> password_reset_success
290
-
291
259
  const locale = this.getLocale(req);
292
- params[pageParams.locale] = locale; // Compose paths and URLs
260
+ params[pageParams.locale] = locale;
293
261
 
262
+ // Compose paths and URLs
294
263
  const defaultFile = page.defaultFile;
295
264
  const defaultPath = this.defaultPagePath(defaultFile);
296
- const defaultUrl = this.composePageUrl(defaultFile, config.publicServerURL); // If custom URL is set redirect to it without localization
265
+ const defaultUrl = this.composePageUrl(defaultFile, config.publicServerURL);
297
266
 
267
+ // If custom URL is set redirect to it without localization
298
268
  const customUrl = config.pages.customUrls[page.id];
299
-
300
269
  if (customUrl && !_Utils.default.isPath(customUrl)) {
301
270
  return this.redirectResponse(customUrl, params);
302
- } // Get JSON placeholders
303
-
271
+ }
304
272
 
273
+ // Get JSON placeholders
305
274
  let placeholders = {};
306
-
307
275
  if (config.pages.enableLocalization && config.pages.localizationJsonPath) {
308
276
  placeholders = this.getJsonPlaceholders(locale, params);
309
- } // Send response
310
-
277
+ }
311
278
 
279
+ // Send response
312
280
  if (config.pages.enableLocalization && locale) {
313
281
  return _Utils.default.getLocalizedPath(defaultPath, locale).then(({
314
282
  path,
@@ -318,37 +286,37 @@ class PagesRouter extends _PromiseRouter.default {
318
286
  return redirect ? this.redirectResponse(defaultUrl, params) : this.pageResponse(defaultPath, params, placeholders);
319
287
  }
320
288
  }
289
+
321
290
  /**
322
291
  * Serves a request to a static resource and localizes the resource if it
323
292
  * is a HTML file.
324
293
  * @param {Object} req The request object.
325
294
  * @returns {Promise<Object>} The response.
326
295
  */
327
-
328
-
329
296
  staticRoute(req) {
330
297
  // Get requested path
331
- const relativePath = req.params[0]; // Resolve requested path to absolute path
332
-
333
- const absolutePath = _path.default.resolve(this.pagesPath, relativePath); // If the requested file is not a HTML file send its raw content
298
+ const relativePath = req.params[0];
334
299
 
300
+ // Resolve requested path to absolute path
301
+ const absolutePath = _path.default.resolve(this.pagesPath, relativePath);
335
302
 
303
+ // If the requested file is not a HTML file send its raw content
336
304
  if (!absolutePath || !absolutePath.endsWith('.html')) {
337
305
  return this.fileResponse(absolutePath);
338
- } // Get parameters
339
-
306
+ }
340
307
 
308
+ // Get parameters
341
309
  const params = this.getDefaultParams(req.config);
342
310
  const locale = this.getLocale(req);
343
-
344
311
  if (locale) {
345
312
  params.locale = locale;
346
- } // Get JSON placeholders
347
-
313
+ }
348
314
 
315
+ // Get JSON placeholders
349
316
  const placeholders = this.getJsonPlaceholders(locale, params);
350
317
  return this.pageResponse(absolutePath, params, placeholders);
351
318
  }
319
+
352
320
  /**
353
321
  * Returns a translation from the JSON resource for a given locale. The JSON
354
322
  * resource is parsed according to i18next syntax.
@@ -377,22 +345,22 @@ class PagesRouter extends _PromiseRouter.default {
377
345
  * @returns {Object} The translation or an empty object if no matching
378
346
  * translation was found.
379
347
  */
380
-
381
-
382
348
  getJsonTranslation(locale) {
383
349
  // If there is no JSON resource
384
350
  if (this.jsonParameters === undefined) {
385
351
  return {};
386
- } // If locale is not set use the fallback locale
387
-
352
+ }
388
353
 
389
- locale = locale || this.pagesConfig.localizationFallbackLocale; // Get matching translation by locale, language or fallback locale
354
+ // If locale is not set use the fallback locale
355
+ locale = locale || this.pagesConfig.localizationFallbackLocale;
390
356
 
357
+ // Get matching translation by locale, language or fallback locale
391
358
  const language = locale.split('-')[0];
392
359
  const resource = this.jsonParameters[locale] || this.jsonParameters[language] || this.jsonParameters[this.pagesConfig.localizationFallbackLocale] || {};
393
360
  const translation = resource.translation || {};
394
361
  return translation;
395
362
  }
363
+
396
364
  /**
397
365
  * Returns a translation from the JSON resource for a given locale with
398
366
  * placeholders filled in by given parameters.
@@ -402,23 +370,23 @@ class PagesRouter extends _PromiseRouter.default {
402
370
  * @returns {Object} The translation or an empty object if no matching
403
371
  * translation was found.
404
372
  */
405
-
406
-
407
373
  getJsonPlaceholders(locale, params = {}) {
408
374
  // If localization is disabled or there is no JSON resource
409
375
  if (!this.pagesConfig.enableLocalization || !this.pagesConfig.localizationJsonPath) {
410
376
  return {};
411
- } // Get JSON placeholders
377
+ }
412
378
 
379
+ // Get JSON placeholders
380
+ let placeholders = this.getJsonTranslation(locale);
413
381
 
414
- let placeholders = this.getJsonTranslation(locale); // Fill in any placeholders in the translation; this allows a translation
382
+ // Fill in any placeholders in the translation; this allows a translation
415
383
  // to contain default placeholders like {{appName}} which are filled here
416
-
417
384
  placeholders = JSON.stringify(placeholders);
418
385
  placeholders = _mustache.default.render(placeholders, params);
419
386
  placeholders = JSON.parse(placeholders);
420
387
  return placeholders;
421
388
  }
389
+
422
390
  /**
423
391
  * Creates a response with file content.
424
392
  * @param {String} path The path of the file to return.
@@ -428,36 +396,32 @@ class PagesRouter extends _PromiseRouter.default {
428
396
  * These will not be included in the response header.
429
397
  * @returns {Object} The Promise Router response.
430
398
  */
431
-
432
-
433
399
  async pageResponse(path, params = {}, placeholders = {}) {
434
400
  // Get file content
435
401
  let data;
436
-
437
402
  try {
438
403
  data = await this.readFile(path);
439
404
  } catch (e) {
440
405
  return this.notFound();
441
- } // Get config placeholders; can be an object, a function or an async function
442
-
406
+ }
443
407
 
408
+ // Get config placeholders; can be an object, a function or an async function
444
409
  let configPlaceholders = typeof this.pagesConfig.placeholders === 'function' ? this.pagesConfig.placeholders(params) : Object.prototype.toString.call(this.pagesConfig.placeholders) === '[object Object]' ? this.pagesConfig.placeholders : {};
445
-
446
410
  if (configPlaceholders instanceof Promise) {
447
411
  configPlaceholders = await configPlaceholders;
448
- } // Fill placeholders
449
-
412
+ }
450
413
 
414
+ // Fill placeholders
451
415
  const allPlaceholders = Object.assign({}, configPlaceholders, placeholders);
452
416
  const paramsAndPlaceholders = Object.assign({}, params, allPlaceholders);
453
- data = _mustache.default.render(data, paramsAndPlaceholders); // Add placeholders in header to allow parsing for programmatic use
454
- // of response, instead of having to parse the HTML content.
417
+ data = _mustache.default.render(data, paramsAndPlaceholders);
455
418
 
419
+ // Add placeholders in header to allow parsing for programmatic use
420
+ // of response, instead of having to parse the HTML content.
456
421
  const headers = Object.entries(params).reduce((m, p) => {
457
422
  if (p[1] !== undefined) {
458
423
  m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];
459
424
  }
460
-
461
425
  return m;
462
426
  }, {});
463
427
  return {
@@ -465,27 +429,25 @@ class PagesRouter extends _PromiseRouter.default {
465
429
  headers: headers
466
430
  };
467
431
  }
432
+
468
433
  /**
469
434
  * Creates a response with file content.
470
435
  * @param {String} path The path of the file to return.
471
436
  * @returns {Object} The PromiseRouter response.
472
437
  */
473
-
474
-
475
438
  async fileResponse(path) {
476
439
  // Get file content
477
440
  let data;
478
-
479
441
  try {
480
442
  data = await this.readFile(path);
481
443
  } catch (e) {
482
444
  return this.notFound();
483
445
  }
484
-
485
446
  return {
486
447
  text: data
487
448
  };
488
449
  }
450
+
489
451
  /**
490
452
  * Reads and returns the content of a file at a given path. File reading to
491
453
  * serve content on the static route is only allowed from the pages
@@ -497,40 +459,35 @@ class PagesRouter extends _PromiseRouter.default {
497
459
  * @param {String} filePath The path to the file to read.
498
460
  * @returns {Promise<String>} The file content.
499
461
  */
500
-
501
-
502
462
  async readFile(filePath) {
503
463
  // Normalize path to prevent it from containing any directory changing
504
464
  // UNIX patterns which could expose the whole file system, e.g.
505
465
  // `http://example.com/parse/apps/../file.txt` requests a file outside
506
466
  // of the pages directory scope.
507
- const normalizedPath = _path.default.normalize(filePath); // Abort if the path is outside of the path directory scope
508
-
467
+ const normalizedPath = _path.default.normalize(filePath);
509
468
 
469
+ // Abort if the path is outside of the path directory scope
510
470
  if (!normalizedPath.startsWith(this.pagesPath)) {
511
471
  throw errors.fileOutsideAllowedScope;
512
472
  }
513
-
514
473
  return await _fs.promises.readFile(normalizedPath, 'utf-8');
515
474
  }
475
+
516
476
  /**
517
477
  * Loads a language resource JSON file that is used for translations.
518
478
  */
519
-
520
-
521
479
  loadJsonResource() {
522
480
  if (this.pagesConfig.localizationJsonPath === undefined) {
523
481
  return;
524
482
  }
525
-
526
483
  try {
527
484
  const json = require(_path.default.resolve('./', this.pagesConfig.localizationJsonPath));
528
-
529
485
  this.jsonParameters = json;
530
486
  } catch (e) {
531
487
  throw errors.jsonFailedFileLoading;
532
488
  }
533
489
  }
490
+
534
491
  /**
535
492
  * Extracts and returns the page default parameters from the Parse Server
536
493
  * configuration. These parameters are made accessible in every page served
@@ -538,8 +495,6 @@ class PagesRouter extends _PromiseRouter.default {
538
495
  * @param {Object} config The Parse Server configuration.
539
496
  * @returns {Object} The default parameters.
540
497
  */
541
-
542
-
543
498
  getDefaultParams(config) {
544
499
  return config ? {
545
500
  [pageParams.appId]: config.appId,
@@ -547,17 +502,17 @@ class PagesRouter extends _PromiseRouter.default {
547
502
  [pageParams.publicServerUrl]: config.publicServerURL
548
503
  } : {};
549
504
  }
505
+
550
506
  /**
551
507
  * Extracts and returns the locale from an express request.
552
508
  * @param {Object} req The express request.
553
509
  * @returns {String|undefined} The locale, or undefined if no locale was set.
554
510
  */
555
-
556
-
557
511
  getLocale(req) {
558
512
  const locale = (req.query || {})[pageParams.locale] || (req.body || {})[pageParams.locale] || (req.params || {})[pageParams.locale] || (req.headers || {})[pageParamHeaderPrefix + pageParams.locale];
559
513
  return locale;
560
514
  }
515
+
561
516
  /**
562
517
  * Creates a response with http redirect.
563
518
  * @param {Object} req The express request.
@@ -565,28 +520,26 @@ class PagesRouter extends _PromiseRouter.default {
565
520
  * @param {Object} params The query parameters to include.
566
521
  * @returns {Object} The Promise Router response.
567
522
  */
568
-
569
-
570
523
  async redirectResponse(url, params) {
571
524
  // Remove any parameters with undefined value
572
525
  params = Object.entries(params).reduce((m, p) => {
573
526
  if (p[1] !== undefined) {
574
527
  m[p[0]] = p[1];
575
528
  }
576
-
577
529
  return m;
578
- }, {}); // Compose URL with parameters in query
530
+ }, {});
579
531
 
532
+ // Compose URL with parameters in query
580
533
  const location = new URL(url);
581
534
  Object.entries(params).forEach(p => location.searchParams.set(p[0], p[1]));
582
- const locationString = location.toString(); // Add parameters to header to allow parsing for programmatic use
583
- // of response, instead of having to parse the HTML content.
535
+ const locationString = location.toString();
584
536
 
537
+ // Add parameters to header to allow parsing for programmatic use
538
+ // of response, instead of having to parse the HTML content.
585
539
  const headers = Object.entries(params).reduce((m, p) => {
586
540
  if (p[1] !== undefined) {
587
541
  m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];
588
542
  }
589
-
590
543
  return m;
591
544
  }, {});
592
545
  return {
@@ -595,11 +548,9 @@ class PagesRouter extends _PromiseRouter.default {
595
548
  headers: headers
596
549
  };
597
550
  }
598
-
599
551
  defaultPagePath(file) {
600
552
  return _path.default.join(this.pagesPath, file);
601
553
  }
602
-
603
554
  composePageUrl(file, publicServerUrl, locale) {
604
555
  let url = publicServerUrl;
605
556
  url += url.endsWith('/') ? '' : '/';
@@ -608,20 +559,19 @@ class PagesRouter extends _PromiseRouter.default {
608
559
  url += file;
609
560
  return url;
610
561
  }
611
-
612
562
  notFound() {
613
563
  return {
614
564
  text: 'Not found.',
615
565
  status: 404
616
566
  };
617
567
  }
618
-
619
568
  invalidRequest() {
620
569
  const error = new Error();
621
570
  error.status = 403;
622
571
  error.message = 'unauthorized';
623
572
  throw error;
624
573
  }
574
+
625
575
  /**
626
576
  * Sets the Parse Server configuration in the request object to make it
627
577
  * easily accessible throughtout request processing.
@@ -629,18 +579,13 @@ class PagesRouter extends _PromiseRouter.default {
629
579
  * @param {Boolean} failGracefully Is true if failing to set the config should
630
580
  * not result in an invalid request response. Default is `false`.
631
581
  */
632
-
633
-
634
582
  setConfig(req, failGracefully = false) {
635
583
  req.config = _Config.default.get(req.params.appId || req.query.appId);
636
-
637
584
  if (!req.config && !failGracefully) {
638
585
  this.invalidRequest();
639
586
  }
640
-
641
587
  return Promise.resolve();
642
588
  }
643
-
644
589
  mountPagesRoutes() {
645
590
  this.route('GET', `/${this.pagesEndpoint}/:appId/verify_email`, req => {
646
591
  this.setConfig(req);
@@ -668,7 +613,6 @@ class PagesRouter extends _PromiseRouter.default {
668
613
  return this.requestResetPassword(req);
669
614
  });
670
615
  }
671
-
672
616
  mountCustomRoutes() {
673
617
  for (const route of this.pagesConfig.customRoutes || []) {
674
618
  this.route(route.method, `/${this.pagesEndpoint}/:appId/${route.path}`, req => {
@@ -677,13 +621,14 @@ class PagesRouter extends _PromiseRouter.default {
677
621
  const {
678
622
  file,
679
623
  query = {}
680
- } = (await route.handler(req)) || {}; // If route handler did not return a page send 404 response
624
+ } = (await route.handler(req)) || {};
681
625
 
626
+ // If route handler did not return a page send 404 response
682
627
  if (!file) {
683
628
  return this.notFound();
684
- } // Send page response
685
-
629
+ }
686
630
 
631
+ // Send page response
687
632
  const page = new _Page.default({
688
633
  id: file,
689
634
  defaultFile: file
@@ -692,7 +637,6 @@ class PagesRouter extends _PromiseRouter.default {
692
637
  });
693
638
  }
694
639
  }
695
-
696
640
  mountStaticRoute() {
697
641
  this.route('GET', `/${this.pagesEndpoint}/(*)?`, req => {
698
642
  this.setConfig(req, true);
@@ -700,16 +644,12 @@ class PagesRouter extends _PromiseRouter.default {
700
644
  return this.staticRoute(req);
701
645
  });
702
646
  }
703
-
704
647
  expressRouter() {
705
648
  const router = _express.default.Router();
706
-
707
649
  router.use('/', super.expressRouter());
708
650
  return router;
709
651
  }
710
-
711
652
  }
712
-
713
653
  exports.PagesRouter = PagesRouter;
714
654
  var _default = PagesRouter;
715
655
  exports.default = _default;
@@ -719,4 +659,4 @@ module.exports = {
719
659
  pageParams,
720
660
  pages
721
661
  };
722
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Routers/PagesRouter.js"],"names":["pages","Object","freeze","passwordReset","Page","id","defaultFile","passwordResetSuccess","passwordResetLinkInvalid","emailVerificationSuccess","emailVerificationSendFail","emailVerificationSendSuccess","emailVerificationLinkInvalid","emailVerificationLinkExpired","pageParams","appName","appId","token","username","error","locale","publicServerUrl","pageParamHeaderPrefix","errors","jsonFailedFileLoading","fileOutsideAllowedScope","PagesRouter","PromiseRouter","constructor","pagesConfig","pagesEndpoint","pagesPath","path","resolve","__dirname","loadJsonResource","mountPagesRoutes","mountCustomRoutes","mountStaticRoute","verifyEmail","req","config","rawToken","query","toString","invalidRequest","goToPage","userController","then","params","resendVerificationEmail","body","publicServerURL","requestResetPassword","checkResetTokenValidity","applicationId","resetPassword","new_password","xhr","Parse","Error","USERNAME_MISSING","OTHER_CAUSE","PASSWORD_MISSING","updatePassword","Promise","success","err","result","status","response","page","responseType","redirect","forceRedirect","undefined","method","defaultParams","getDefaultParams","values","includes","notFound","assign","getLocale","defaultPath","defaultPagePath","defaultUrl","composePageUrl","customUrl","customUrls","Utils","isPath","redirectResponse","placeholders","enableLocalization","localizationJsonPath","getJsonPlaceholders","getLocalizedPath","subdir","pageResponse","staticRoute","relativePath","absolutePath","endsWith","fileResponse","getJsonTranslation","jsonParameters","localizationFallbackLocale","language","split","resource","translation","JSON","stringify","mustache","render","parse","data","readFile","e","configPlaceholders","prototype","call","allPlaceholders","paramsAndPlaceholders","headers","entries","reduce","m","p","toLowerCase","text","filePath","normalizedPath","normalize","startsWith","fs","json","require","url","location","URL","forEach","searchParams","set","locationString","file","join","message","setConfig","failGracefully","Config","get","route","customRoutes","handler","expressRouter","router","express","Router","use","module","exports"],"mappings":"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","sourcesContent":["import PromiseRouter from '../PromiseRouter';\nimport Config from '../Config';\nimport express from 'express';\nimport path from 'path';\nimport { promises as fs } from 'fs';\nimport { Parse } from 'parse/node';\nimport Utils from '../Utils';\nimport mustache from 'mustache';\nimport Page from '../Page';\n\n// All pages with custom page key for reference and file name\nconst pages = Object.freeze({\n  passwordReset: new Page({ id: 'passwordReset', defaultFile: 'password_reset.html' }),\n  passwordResetSuccess: new Page({\n    id: 'passwordResetSuccess',\n    defaultFile: 'password_reset_success.html',\n  }),\n  passwordResetLinkInvalid: new Page({\n    id: 'passwordResetLinkInvalid',\n    defaultFile: 'password_reset_link_invalid.html',\n  }),\n  emailVerificationSuccess: new Page({\n    id: 'emailVerificationSuccess',\n    defaultFile: 'email_verification_success.html',\n  }),\n  emailVerificationSendFail: new Page({\n    id: 'emailVerificationSendFail',\n    defaultFile: 'email_verification_send_fail.html',\n  }),\n  emailVerificationSendSuccess: new Page({\n    id: 'emailVerificationSendSuccess',\n    defaultFile: 'email_verification_send_success.html',\n  }),\n  emailVerificationLinkInvalid: new Page({\n    id: 'emailVerificationLinkInvalid',\n    defaultFile: 'email_verification_link_invalid.html',\n  }),\n  emailVerificationLinkExpired: new Page({\n    id: 'emailVerificationLinkExpired',\n    defaultFile: 'email_verification_link_expired.html',\n  }),\n});\n\n// All page parameters for reference to be used as template placeholders or query params\nconst pageParams = Object.freeze({\n  appName: 'appName',\n  appId: 'appId',\n  token: 'token',\n  username: 'username',\n  error: 'error',\n  locale: 'locale',\n  publicServerUrl: 'publicServerUrl',\n});\n\n// The header prefix to add page params as response headers\nconst pageParamHeaderPrefix = 'x-parse-page-param-';\n\n// The errors being thrown\nconst errors = Object.freeze({\n  jsonFailedFileLoading: 'failed to load JSON file',\n  fileOutsideAllowedScope: 'not allowed to read file outside of pages directory',\n});\n\nexport class PagesRouter extends PromiseRouter {\n  /**\n   * Constructs a PagesRouter.\n   * @param {Object} pages The pages options from the Parse Server configuration.\n   */\n  constructor(pages = {}) {\n    super();\n\n    // Set instance properties\n    this.pagesConfig = pages;\n    this.pagesEndpoint = pages.pagesEndpoint ? pages.pagesEndpoint : 'apps';\n    this.pagesPath = pages.pagesPath\n      ? path.resolve('./', pages.pagesPath)\n      : path.resolve(__dirname, '../../public');\n    this.loadJsonResource();\n    this.mountPagesRoutes();\n    this.mountCustomRoutes();\n    this.mountStaticRoute();\n  }\n\n  verifyEmail(req) {\n    const config = req.config;\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!token || !username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n    return userController.verifyEmail(username, token).then(\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationSuccess, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationLinkExpired, params);\n      }\n    );\n  }\n\n  resendVerificationEmail(req) {\n    const config = req.config;\n    const username = req.body.username;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n\n    return userController.resendVerificationEmail(username).then(\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendSuccess);\n      },\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendFail);\n      }\n    );\n  }\n\n  passwordReset(req) {\n    const config = req.config;\n    const params = {\n      [pageParams.appId]: req.params.appId,\n      [pageParams.appName]: config.appName,\n      [pageParams.token]: req.query.token,\n      [pageParams.username]: req.query.username,\n      [pageParams.publicServerUrl]: config.publicServerURL,\n    };\n    return this.goToPage(req, pages.passwordReset, params);\n  }\n\n  requestResetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!username || !token) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    return config.userController.checkResetTokenValidity(username, token).then(\n      () => {\n        const params = {\n          [pageParams.token]: token,\n          [pageParams.username]: username,\n          [pageParams.appId]: config.applicationId,\n          [pageParams.appName]: config.appName,\n        };\n        return this.goToPage(req, pages.passwordReset, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.passwordResetLinkInvalid, params);\n      }\n    );\n  }\n\n  resetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, new_password, token: rawToken } = req.body;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if ((!username || !token || !new_password) && req.xhr === false) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    if (!username) {\n      throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'Missing username');\n    }\n\n    if (!token) {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'Missing token');\n    }\n\n    if (!new_password) {\n      throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'Missing password');\n    }\n\n    return config.userController\n      .updatePassword(username, token, new_password)\n      .then(\n        () => {\n          return Promise.resolve({\n            success: true,\n          });\n        },\n        err => {\n          return Promise.resolve({\n            success: false,\n            err,\n          });\n        }\n      )\n      .then(result => {\n        if (req.xhr) {\n          if (result.success) {\n            return Promise.resolve({\n              status: 200,\n              response: 'Password successfully reset',\n            });\n          }\n          if (result.err) {\n            throw new Parse.Error(Parse.Error.OTHER_CAUSE, `${result.err}`);\n          }\n        }\n\n        const query = result.success\n          ? {\n            [pageParams.username]: username,\n          }\n          : {\n            [pageParams.username]: username,\n            [pageParams.token]: token,\n            [pageParams.appId]: config.applicationId,\n            [pageParams.error]: result.err,\n            [pageParams.appName]: config.appName,\n          };\n        const page = result.success ? pages.passwordResetSuccess : pages.passwordReset;\n\n        return this.goToPage(req, page, query, false);\n      });\n  }\n\n  /**\n   * Returns page content if the page is a local file or returns a\n   * redirect to a custom page.\n   * @param {Object} req The express request.\n   * @param {Page} page The page to go to.\n   * @param {Object} [params={}] The query parameters to attach to the URL in case of\n   * HTTP redirect responses for POST requests, or the placeholders to fill into\n   * the response content in case of HTTP content responses for GET requests.\n   * @param {Boolean} [responseType] Is true if a redirect response should be forced,\n   * false if a content response should be forced, undefined if the response type\n   * should depend on the request type by default:\n   * - GET request -> content response\n   * - POST request -> redirect response (PRG pattern)\n   * @returns {Promise<Object>} The PromiseRouter response.\n   */\n  goToPage(req, page, params = {}, responseType) {\n    const config = req.config;\n\n    // Determine redirect either by force, response setting or request method\n    const redirect = config.pages.forceRedirect\n      ? true\n      : responseType !== undefined\n        ? responseType\n        : req.method == 'POST';\n\n    // Include default parameters\n    const defaultParams = this.getDefaultParams(config);\n    if (Object.values(defaultParams).includes(undefined)) {\n      return this.notFound();\n    }\n    params = Object.assign(params, defaultParams);\n\n    // Add locale to params to ensure it is passed on with every request;\n    // that means, once a locale is set, it is passed on to any follow-up page,\n    // e.g. request_password_reset -> password_reset -> password_reset_success\n    const locale = this.getLocale(req);\n    params[pageParams.locale] = locale;\n\n    // Compose paths and URLs\n    const defaultFile = page.defaultFile;\n    const defaultPath = this.defaultPagePath(defaultFile);\n    const defaultUrl = this.composePageUrl(defaultFile, config.publicServerURL);\n\n    // If custom URL is set redirect to it without localization\n    const customUrl = config.pages.customUrls[page.id];\n    if (customUrl && !Utils.isPath(customUrl)) {\n      return this.redirectResponse(customUrl, params);\n    }\n\n    // Get JSON placeholders\n    let placeholders = {};\n    if (config.pages.enableLocalization && config.pages.localizationJsonPath) {\n      placeholders = this.getJsonPlaceholders(locale, params);\n    }\n\n    // Send response\n    if (config.pages.enableLocalization && locale) {\n      return Utils.getLocalizedPath(defaultPath, locale).then(({ path, subdir }) =>\n        redirect\n          ? this.redirectResponse(\n            this.composePageUrl(defaultFile, config.publicServerURL, subdir),\n            params\n          )\n          : this.pageResponse(path, params, placeholders)\n      );\n    } else {\n      return redirect\n        ? this.redirectResponse(defaultUrl, params)\n        : this.pageResponse(defaultPath, params, placeholders);\n    }\n  }\n\n  /**\n   * Serves a request to a static resource and localizes the resource if it\n   * is a HTML file.\n   * @param {Object} req The request object.\n   * @returns {Promise<Object>} The response.\n   */\n  staticRoute(req) {\n    // Get requested path\n    const relativePath = req.params[0];\n\n    // Resolve requested path to absolute path\n    const absolutePath = path.resolve(this.pagesPath, relativePath);\n\n    // If the requested file is not a HTML file send its raw content\n    if (!absolutePath || !absolutePath.endsWith('.html')) {\n      return this.fileResponse(absolutePath);\n    }\n\n    // Get parameters\n    const params = this.getDefaultParams(req.config);\n    const locale = this.getLocale(req);\n    if (locale) {\n      params.locale = locale;\n    }\n\n    // Get JSON placeholders\n    const placeholders = this.getJsonPlaceholders(locale, params);\n\n    return this.pageResponse(absolutePath, params, placeholders);\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale. The JSON\n   * resource is parsed according to i18next syntax.\n   *\n   * Example JSON content:\n   * ```js\n   *  {\n   *    \"en\": {               // resource for language `en` (English)\n   *      \"translation\": {\n   *        \"greeting\": \"Hello!\"\n   *      }\n   *    },\n   *    \"de\": {               // resource for language `de` (German)\n   *      \"translation\": {\n   *        \"greeting\": \"Hallo!\"\n   *      }\n   *    }\n   *    \"de-CH\": {            // resource for locale `de-CH` (Swiss German)\n   *      \"translation\": {\n   *        \"greeting\": \"Grüezi!\"\n   *      }\n   *    }\n   *  }\n   * ```\n   * @param {String} locale The locale to translate to.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonTranslation(locale) {\n    // If there is no JSON resource\n    if (this.jsonParameters === undefined) {\n      return {};\n    }\n\n    // If locale is not set use the fallback locale\n    locale = locale || this.pagesConfig.localizationFallbackLocale;\n\n    // Get matching translation by locale, language or fallback locale\n    const language = locale.split('-')[0];\n    const resource =\n      this.jsonParameters[locale] ||\n      this.jsonParameters[language] ||\n      this.jsonParameters[this.pagesConfig.localizationFallbackLocale] ||\n      {};\n    const translation = resource.translation || {};\n    return translation;\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale with\n   * placeholders filled in by given parameters.\n   * @param {String} locale The locale to translate to.\n   * @param {Object} params The parameters to fill into any placeholders\n   * within the translations.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonPlaceholders(locale, params = {}) {\n    // If localization is disabled or there is no JSON resource\n    if (!this.pagesConfig.enableLocalization || !this.pagesConfig.localizationJsonPath) {\n      return {};\n    }\n\n    // Get JSON placeholders\n    let placeholders = this.getJsonTranslation(locale);\n\n    // Fill in any placeholders in the translation; this allows a translation\n    // to contain default placeholders like {{appName}} which are filled here\n    placeholders = JSON.stringify(placeholders);\n    placeholders = mustache.render(placeholders, params);\n    placeholders = JSON.parse(placeholders);\n\n    return placeholders;\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @param {Object} [params={}] The parameters to be included in the response\n   * header. These will also be used to fill placeholders.\n   * @param {Object} [placeholders={}] The placeholders to fill in the content.\n   * These will not be included in the response header.\n   * @returns {Object} The Promise Router response.\n   */\n  async pageResponse(path, params = {}, placeholders = {}) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    // Get config placeholders; can be an object, a function or an async function\n    let configPlaceholders =\n      typeof this.pagesConfig.placeholders === 'function'\n        ? this.pagesConfig.placeholders(params)\n        : Object.prototype.toString.call(this.pagesConfig.placeholders) === '[object Object]'\n          ? this.pagesConfig.placeholders\n          : {};\n    if (configPlaceholders instanceof Promise) {\n      configPlaceholders = await configPlaceholders;\n    }\n\n    // Fill placeholders\n    const allPlaceholders = Object.assign({}, configPlaceholders, placeholders);\n    const paramsAndPlaceholders = Object.assign({}, params, allPlaceholders);\n    data = mustache.render(data, paramsAndPlaceholders);\n\n    // Add placeholders in header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return { text: data, headers: headers };\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @returns {Object} The PromiseRouter response.\n   */\n  async fileResponse(path) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    return { text: data };\n  }\n\n  /**\n   * Reads and returns the content of a file at a given path. File reading to\n   * serve content on the static route is only allowed from the pages\n   * directory on downwards.\n   * -----------------------------------------------------------------------\n   * **WARNING:** All file reads in the PagesRouter must be executed by this\n   * wrapper because it also detects and prevents common exploits.\n   * -----------------------------------------------------------------------\n   * @param {String} filePath The path to the file to read.\n   * @returns {Promise<String>} The file content.\n   */\n  async readFile(filePath) {\n    // Normalize path to prevent it from containing any directory changing\n    // UNIX patterns which could expose the whole file system, e.g.\n    // `http://example.com/parse/apps/../file.txt` requests a file outside\n    // of the pages directory scope.\n    const normalizedPath = path.normalize(filePath);\n\n    // Abort if the path is outside of the path directory scope\n    if (!normalizedPath.startsWith(this.pagesPath)) {\n      throw errors.fileOutsideAllowedScope;\n    }\n\n    return await fs.readFile(normalizedPath, 'utf-8');\n  }\n\n  /**\n   * Loads a language resource JSON file that is used for translations.\n   */\n  loadJsonResource() {\n    if (this.pagesConfig.localizationJsonPath === undefined) {\n      return;\n    }\n    try {\n      const json = require(path.resolve('./', this.pagesConfig.localizationJsonPath));\n      this.jsonParameters = json;\n    } catch (e) {\n      throw errors.jsonFailedFileLoading;\n    }\n  }\n\n  /**\n   * Extracts and returns the page default parameters from the Parse Server\n   * configuration. These parameters are made accessible in every page served\n   * by this router.\n   * @param {Object} config The Parse Server configuration.\n   * @returns {Object} The default parameters.\n   */\n  getDefaultParams(config) {\n    return config\n      ? {\n        [pageParams.appId]: config.appId,\n        [pageParams.appName]: config.appName,\n        [pageParams.publicServerUrl]: config.publicServerURL,\n      }\n      : {};\n  }\n\n  /**\n   * Extracts and returns the locale from an express request.\n   * @param {Object} req The express request.\n   * @returns {String|undefined} The locale, or undefined if no locale was set.\n   */\n  getLocale(req) {\n    const locale =\n      (req.query || {})[pageParams.locale] ||\n      (req.body || {})[pageParams.locale] ||\n      (req.params || {})[pageParams.locale] ||\n      (req.headers || {})[pageParamHeaderPrefix + pageParams.locale];\n    return locale;\n  }\n\n  /**\n   * Creates a response with http redirect.\n   * @param {Object} req The express request.\n   * @param {String} path The path of the file to return.\n   * @param {Object} params The query parameters to include.\n   * @returns {Object} The Promise Router response.\n   */\n  async redirectResponse(url, params) {\n    // Remove any parameters with undefined value\n    params = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[p[0]] = p[1];\n      }\n      return m;\n    }, {});\n\n    // Compose URL with parameters in query\n    const location = new URL(url);\n    Object.entries(params).forEach(p => location.searchParams.set(p[0], p[1]));\n    const locationString = location.toString();\n\n    // Add parameters to header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return {\n      status: 303,\n      location: locationString,\n      headers: headers,\n    };\n  }\n\n  defaultPagePath(file) {\n    return path.join(this.pagesPath, file);\n  }\n\n  composePageUrl(file, publicServerUrl, locale) {\n    let url = publicServerUrl;\n    url += url.endsWith('/') ? '' : '/';\n    url += this.pagesEndpoint + '/';\n    url += locale === undefined ? '' : locale + '/';\n    url += file;\n    return url;\n  }\n\n  notFound() {\n    return {\n      text: 'Not found.',\n      status: 404,\n    };\n  }\n\n  invalidRequest() {\n    const error = new Error();\n    error.status = 403;\n    error.message = 'unauthorized';\n    throw error;\n  }\n\n  /**\n   * Sets the Parse Server configuration in the request object to make it\n   * easily accessible throughtout request processing.\n   * @param {Object} req The request.\n   * @param {Boolean} failGracefully Is true if failing to set the config should\n   * not result in an invalid request response. Default is `false`.\n   */\n  setConfig(req, failGracefully = false) {\n    req.config = Config.get(req.params.appId || req.query.appId);\n    if (!req.config && !failGracefully) {\n      this.invalidRequest();\n    }\n    return Promise.resolve();\n  }\n\n  mountPagesRoutes() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/verify_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.verifyEmail(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/resend_verification_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resendVerificationEmail(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/choose_password`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.passwordReset(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resetPassword(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.requestResetPassword(req);\n      }\n    );\n  }\n\n  mountCustomRoutes() {\n    for (const route of this.pagesConfig.customRoutes || []) {\n      this.route(\n        route.method,\n        `/${this.pagesEndpoint}/:appId/${route.path}`,\n        req => {\n          this.setConfig(req);\n        },\n        async req => {\n          const { file, query = {} } = (await route.handler(req)) || {};\n\n          // If route handler did not return a page send 404 response\n          if (!file) {\n            return this.notFound();\n          }\n\n          // Send page response\n          const page = new Page({ id: file, defaultFile: file });\n          return this.goToPage(req, page, query, false);\n        }\n      );\n    }\n  }\n\n  mountStaticRoute() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/(*)?`,\n      req => {\n        this.setConfig(req, true);\n      },\n      req => {\n        return this.staticRoute(req);\n      }\n    );\n  }\n\n  expressRouter() {\n    const router = express.Router();\n    router.use('/', super.expressRouter());\n    return router;\n  }\n}\n\nexport default PagesRouter;\nmodule.exports = {\n  PagesRouter,\n  pageParamHeaderPrefix,\n  pageParams,\n  pages,\n};\n"]}
662
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["pages","Object","freeze","passwordReset","Page","id","defaultFile","passwordResetSuccess","passwordResetLinkInvalid","emailVerificationSuccess","emailVerificationSendFail","emailVerificationSendSuccess","emailVerificationLinkInvalid","emailVerificationLinkExpired","pageParams","appName","appId","token","username","error","locale","publicServerUrl","pageParamHeaderPrefix","errors","jsonFailedFileLoading","fileOutsideAllowedScope","PagesRouter","PromiseRouter","constructor","pagesConfig","pagesEndpoint","pagesPath","path","resolve","__dirname","loadJsonResource","mountPagesRoutes","mountCustomRoutes","mountStaticRoute","verifyEmail","req","config","rawToken","query","toString","invalidRequest","goToPage","userController","then","params","resendVerificationEmail","body","publicServerURL","requestResetPassword","checkResetTokenValidity","applicationId","resetPassword","new_password","xhr","Parse","Error","USERNAME_MISSING","OTHER_CAUSE","PASSWORD_MISSING","updatePassword","Promise","success","err","result","status","response","page","responseType","redirect","forceRedirect","undefined","method","defaultParams","getDefaultParams","values","includes","notFound","assign","getLocale","defaultPath","defaultPagePath","defaultUrl","composePageUrl","customUrl","customUrls","Utils","isPath","redirectResponse","placeholders","enableLocalization","localizationJsonPath","getJsonPlaceholders","getLocalizedPath","subdir","pageResponse","staticRoute","relativePath","absolutePath","endsWith","fileResponse","getJsonTranslation","jsonParameters","localizationFallbackLocale","language","split","resource","translation","JSON","stringify","mustache","render","parse","data","readFile","e","configPlaceholders","prototype","call","allPlaceholders","paramsAndPlaceholders","headers","entries","reduce","m","p","toLowerCase","text","filePath","normalizedPath","normalize","startsWith","fs","json","require","url","location","URL","forEach","searchParams","set","locationString","file","join","message","setConfig","failGracefully","Config","get","route","customRoutes","handler","expressRouter","router","express","Router","use","module","exports"],"sources":["../../src/Routers/PagesRouter.js"],"sourcesContent":["import PromiseRouter from '../PromiseRouter';\nimport Config from '../Config';\nimport express from 'express';\nimport path from 'path';\nimport { promises as fs } from 'fs';\nimport { Parse } from 'parse/node';\nimport Utils from '../Utils';\nimport mustache from 'mustache';\nimport Page from '../Page';\n\n// All pages with custom page key for reference and file name\nconst pages = Object.freeze({\n  passwordReset: new Page({ id: 'passwordReset', defaultFile: 'password_reset.html' }),\n  passwordResetSuccess: new Page({\n    id: 'passwordResetSuccess',\n    defaultFile: 'password_reset_success.html',\n  }),\n  passwordResetLinkInvalid: new Page({\n    id: 'passwordResetLinkInvalid',\n    defaultFile: 'password_reset_link_invalid.html',\n  }),\n  emailVerificationSuccess: new Page({\n    id: 'emailVerificationSuccess',\n    defaultFile: 'email_verification_success.html',\n  }),\n  emailVerificationSendFail: new Page({\n    id: 'emailVerificationSendFail',\n    defaultFile: 'email_verification_send_fail.html',\n  }),\n  emailVerificationSendSuccess: new Page({\n    id: 'emailVerificationSendSuccess',\n    defaultFile: 'email_verification_send_success.html',\n  }),\n  emailVerificationLinkInvalid: new Page({\n    id: 'emailVerificationLinkInvalid',\n    defaultFile: 'email_verification_link_invalid.html',\n  }),\n  emailVerificationLinkExpired: new Page({\n    id: 'emailVerificationLinkExpired',\n    defaultFile: 'email_verification_link_expired.html',\n  }),\n});\n\n// All page parameters for reference to be used as template placeholders or query params\nconst pageParams = Object.freeze({\n  appName: 'appName',\n  appId: 'appId',\n  token: 'token',\n  username: 'username',\n  error: 'error',\n  locale: 'locale',\n  publicServerUrl: 'publicServerUrl',\n});\n\n// The header prefix to add page params as response headers\nconst pageParamHeaderPrefix = 'x-parse-page-param-';\n\n// The errors being thrown\nconst errors = Object.freeze({\n  jsonFailedFileLoading: 'failed to load JSON file',\n  fileOutsideAllowedScope: 'not allowed to read file outside of pages directory',\n});\n\nexport class PagesRouter extends PromiseRouter {\n  /**\n   * Constructs a PagesRouter.\n   * @param {Object} pages The pages options from the Parse Server configuration.\n   */\n  constructor(pages = {}) {\n    super();\n\n    // Set instance properties\n    this.pagesConfig = pages;\n    this.pagesEndpoint = pages.pagesEndpoint ? pages.pagesEndpoint : 'apps';\n    this.pagesPath = pages.pagesPath\n      ? path.resolve('./', pages.pagesPath)\n      : path.resolve(__dirname, '../../public');\n    this.loadJsonResource();\n    this.mountPagesRoutes();\n    this.mountCustomRoutes();\n    this.mountStaticRoute();\n  }\n\n  verifyEmail(req) {\n    const config = req.config;\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!token || !username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n    return userController.verifyEmail(username, token).then(\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationSuccess, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.emailVerificationLinkExpired, params);\n      }\n    );\n  }\n\n  resendVerificationEmail(req) {\n    const config = req.config;\n    const username = req.body.username;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    if (!username) {\n      return this.goToPage(req, pages.emailVerificationLinkInvalid);\n    }\n\n    const userController = config.userController;\n\n    return userController.resendVerificationEmail(username).then(\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendSuccess);\n      },\n      () => {\n        return this.goToPage(req, pages.emailVerificationSendFail);\n      }\n    );\n  }\n\n  passwordReset(req) {\n    const config = req.config;\n    const params = {\n      [pageParams.appId]: req.params.appId,\n      [pageParams.appName]: config.appName,\n      [pageParams.token]: req.query.token,\n      [pageParams.username]: req.query.username,\n      [pageParams.publicServerUrl]: config.publicServerURL,\n    };\n    return this.goToPage(req, pages.passwordReset, params);\n  }\n\n  requestResetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, token: rawToken } = req.query;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if (!username || !token) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    return config.userController.checkResetTokenValidity(username, token).then(\n      () => {\n        const params = {\n          [pageParams.token]: token,\n          [pageParams.username]: username,\n          [pageParams.appId]: config.applicationId,\n          [pageParams.appName]: config.appName,\n        };\n        return this.goToPage(req, pages.passwordReset, params);\n      },\n      () => {\n        const params = {\n          [pageParams.username]: username,\n        };\n        return this.goToPage(req, pages.passwordResetLinkInvalid, params);\n      }\n    );\n  }\n\n  resetPassword(req) {\n    const config = req.config;\n\n    if (!config) {\n      this.invalidRequest();\n    }\n\n    const { username, new_password, token: rawToken } = req.body;\n    const token = rawToken && typeof rawToken !== 'string' ? rawToken.toString() : rawToken;\n\n    if ((!username || !token || !new_password) && req.xhr === false) {\n      return this.goToPage(req, pages.passwordResetLinkInvalid);\n    }\n\n    if (!username) {\n      throw new Parse.Error(Parse.Error.USERNAME_MISSING, 'Missing username');\n    }\n\n    if (!token) {\n      throw new Parse.Error(Parse.Error.OTHER_CAUSE, 'Missing token');\n    }\n\n    if (!new_password) {\n      throw new Parse.Error(Parse.Error.PASSWORD_MISSING, 'Missing password');\n    }\n\n    return config.userController\n      .updatePassword(username, token, new_password)\n      .then(\n        () => {\n          return Promise.resolve({\n            success: true,\n          });\n        },\n        err => {\n          return Promise.resolve({\n            success: false,\n            err,\n          });\n        }\n      )\n      .then(result => {\n        if (req.xhr) {\n          if (result.success) {\n            return Promise.resolve({\n              status: 200,\n              response: 'Password successfully reset',\n            });\n          }\n          if (result.err) {\n            throw new Parse.Error(Parse.Error.OTHER_CAUSE, `${result.err}`);\n          }\n        }\n\n        const query = result.success\n          ? {\n            [pageParams.username]: username,\n          }\n          : {\n            [pageParams.username]: username,\n            [pageParams.token]: token,\n            [pageParams.appId]: config.applicationId,\n            [pageParams.error]: result.err,\n            [pageParams.appName]: config.appName,\n          };\n        const page = result.success ? pages.passwordResetSuccess : pages.passwordReset;\n\n        return this.goToPage(req, page, query, false);\n      });\n  }\n\n  /**\n   * Returns page content if the page is a local file or returns a\n   * redirect to a custom page.\n   * @param {Object} req The express request.\n   * @param {Page} page The page to go to.\n   * @param {Object} [params={}] The query parameters to attach to the URL in case of\n   * HTTP redirect responses for POST requests, or the placeholders to fill into\n   * the response content in case of HTTP content responses for GET requests.\n   * @param {Boolean} [responseType] Is true if a redirect response should be forced,\n   * false if a content response should be forced, undefined if the response type\n   * should depend on the request type by default:\n   * - GET request -> content response\n   * - POST request -> redirect response (PRG pattern)\n   * @returns {Promise<Object>} The PromiseRouter response.\n   */\n  goToPage(req, page, params = {}, responseType) {\n    const config = req.config;\n\n    // Determine redirect either by force, response setting or request method\n    const redirect = config.pages.forceRedirect\n      ? true\n      : responseType !== undefined\n        ? responseType\n        : req.method == 'POST';\n\n    // Include default parameters\n    const defaultParams = this.getDefaultParams(config);\n    if (Object.values(defaultParams).includes(undefined)) {\n      return this.notFound();\n    }\n    params = Object.assign(params, defaultParams);\n\n    // Add locale to params to ensure it is passed on with every request;\n    // that means, once a locale is set, it is passed on to any follow-up page,\n    // e.g. request_password_reset -> password_reset -> password_reset_success\n    const locale = this.getLocale(req);\n    params[pageParams.locale] = locale;\n\n    // Compose paths and URLs\n    const defaultFile = page.defaultFile;\n    const defaultPath = this.defaultPagePath(defaultFile);\n    const defaultUrl = this.composePageUrl(defaultFile, config.publicServerURL);\n\n    // If custom URL is set redirect to it without localization\n    const customUrl = config.pages.customUrls[page.id];\n    if (customUrl && !Utils.isPath(customUrl)) {\n      return this.redirectResponse(customUrl, params);\n    }\n\n    // Get JSON placeholders\n    let placeholders = {};\n    if (config.pages.enableLocalization && config.pages.localizationJsonPath) {\n      placeholders = this.getJsonPlaceholders(locale, params);\n    }\n\n    // Send response\n    if (config.pages.enableLocalization && locale) {\n      return Utils.getLocalizedPath(defaultPath, locale).then(({ path, subdir }) =>\n        redirect\n          ? this.redirectResponse(\n            this.composePageUrl(defaultFile, config.publicServerURL, subdir),\n            params\n          )\n          : this.pageResponse(path, params, placeholders)\n      );\n    } else {\n      return redirect\n        ? this.redirectResponse(defaultUrl, params)\n        : this.pageResponse(defaultPath, params, placeholders);\n    }\n  }\n\n  /**\n   * Serves a request to a static resource and localizes the resource if it\n   * is a HTML file.\n   * @param {Object} req The request object.\n   * @returns {Promise<Object>} The response.\n   */\n  staticRoute(req) {\n    // Get requested path\n    const relativePath = req.params[0];\n\n    // Resolve requested path to absolute path\n    const absolutePath = path.resolve(this.pagesPath, relativePath);\n\n    // If the requested file is not a HTML file send its raw content\n    if (!absolutePath || !absolutePath.endsWith('.html')) {\n      return this.fileResponse(absolutePath);\n    }\n\n    // Get parameters\n    const params = this.getDefaultParams(req.config);\n    const locale = this.getLocale(req);\n    if (locale) {\n      params.locale = locale;\n    }\n\n    // Get JSON placeholders\n    const placeholders = this.getJsonPlaceholders(locale, params);\n\n    return this.pageResponse(absolutePath, params, placeholders);\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale. The JSON\n   * resource is parsed according to i18next syntax.\n   *\n   * Example JSON content:\n   * ```js\n   *  {\n   *    \"en\": {               // resource for language `en` (English)\n   *      \"translation\": {\n   *        \"greeting\": \"Hello!\"\n   *      }\n   *    },\n   *    \"de\": {               // resource for language `de` (German)\n   *      \"translation\": {\n   *        \"greeting\": \"Hallo!\"\n   *      }\n   *    }\n   *    \"de-CH\": {            // resource for locale `de-CH` (Swiss German)\n   *      \"translation\": {\n   *        \"greeting\": \"Grüezi!\"\n   *      }\n   *    }\n   *  }\n   * ```\n   * @param {String} locale The locale to translate to.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonTranslation(locale) {\n    // If there is no JSON resource\n    if (this.jsonParameters === undefined) {\n      return {};\n    }\n\n    // If locale is not set use the fallback locale\n    locale = locale || this.pagesConfig.localizationFallbackLocale;\n\n    // Get matching translation by locale, language or fallback locale\n    const language = locale.split('-')[0];\n    const resource =\n      this.jsonParameters[locale] ||\n      this.jsonParameters[language] ||\n      this.jsonParameters[this.pagesConfig.localizationFallbackLocale] ||\n      {};\n    const translation = resource.translation || {};\n    return translation;\n  }\n\n  /**\n   * Returns a translation from the JSON resource for a given locale with\n   * placeholders filled in by given parameters.\n   * @param {String} locale The locale to translate to.\n   * @param {Object} params The parameters to fill into any placeholders\n   * within the translations.\n   * @returns {Object} The translation or an empty object if no matching\n   * translation was found.\n   */\n  getJsonPlaceholders(locale, params = {}) {\n    // If localization is disabled or there is no JSON resource\n    if (!this.pagesConfig.enableLocalization || !this.pagesConfig.localizationJsonPath) {\n      return {};\n    }\n\n    // Get JSON placeholders\n    let placeholders = this.getJsonTranslation(locale);\n\n    // Fill in any placeholders in the translation; this allows a translation\n    // to contain default placeholders like {{appName}} which are filled here\n    placeholders = JSON.stringify(placeholders);\n    placeholders = mustache.render(placeholders, params);\n    placeholders = JSON.parse(placeholders);\n\n    return placeholders;\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @param {Object} [params={}] The parameters to be included in the response\n   * header. These will also be used to fill placeholders.\n   * @param {Object} [placeholders={}] The placeholders to fill in the content.\n   * These will not be included in the response header.\n   * @returns {Object} The Promise Router response.\n   */\n  async pageResponse(path, params = {}, placeholders = {}) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    // Get config placeholders; can be an object, a function or an async function\n    let configPlaceholders =\n      typeof this.pagesConfig.placeholders === 'function'\n        ? this.pagesConfig.placeholders(params)\n        : Object.prototype.toString.call(this.pagesConfig.placeholders) === '[object Object]'\n          ? this.pagesConfig.placeholders\n          : {};\n    if (configPlaceholders instanceof Promise) {\n      configPlaceholders = await configPlaceholders;\n    }\n\n    // Fill placeholders\n    const allPlaceholders = Object.assign({}, configPlaceholders, placeholders);\n    const paramsAndPlaceholders = Object.assign({}, params, allPlaceholders);\n    data = mustache.render(data, paramsAndPlaceholders);\n\n    // Add placeholders in header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return { text: data, headers: headers };\n  }\n\n  /**\n   * Creates a response with file content.\n   * @param {String} path The path of the file to return.\n   * @returns {Object} The PromiseRouter response.\n   */\n  async fileResponse(path) {\n    // Get file content\n    let data;\n    try {\n      data = await this.readFile(path);\n    } catch (e) {\n      return this.notFound();\n    }\n\n    return { text: data };\n  }\n\n  /**\n   * Reads and returns the content of a file at a given path. File reading to\n   * serve content on the static route is only allowed from the pages\n   * directory on downwards.\n   * -----------------------------------------------------------------------\n   * **WARNING:** All file reads in the PagesRouter must be executed by this\n   * wrapper because it also detects and prevents common exploits.\n   * -----------------------------------------------------------------------\n   * @param {String} filePath The path to the file to read.\n   * @returns {Promise<String>} The file content.\n   */\n  async readFile(filePath) {\n    // Normalize path to prevent it from containing any directory changing\n    // UNIX patterns which could expose the whole file system, e.g.\n    // `http://example.com/parse/apps/../file.txt` requests a file outside\n    // of the pages directory scope.\n    const normalizedPath = path.normalize(filePath);\n\n    // Abort if the path is outside of the path directory scope\n    if (!normalizedPath.startsWith(this.pagesPath)) {\n      throw errors.fileOutsideAllowedScope;\n    }\n\n    return await fs.readFile(normalizedPath, 'utf-8');\n  }\n\n  /**\n   * Loads a language resource JSON file that is used for translations.\n   */\n  loadJsonResource() {\n    if (this.pagesConfig.localizationJsonPath === undefined) {\n      return;\n    }\n    try {\n      const json = require(path.resolve('./', this.pagesConfig.localizationJsonPath));\n      this.jsonParameters = json;\n    } catch (e) {\n      throw errors.jsonFailedFileLoading;\n    }\n  }\n\n  /**\n   * Extracts and returns the page default parameters from the Parse Server\n   * configuration. These parameters are made accessible in every page served\n   * by this router.\n   * @param {Object} config The Parse Server configuration.\n   * @returns {Object} The default parameters.\n   */\n  getDefaultParams(config) {\n    return config\n      ? {\n        [pageParams.appId]: config.appId,\n        [pageParams.appName]: config.appName,\n        [pageParams.publicServerUrl]: config.publicServerURL,\n      }\n      : {};\n  }\n\n  /**\n   * Extracts and returns the locale from an express request.\n   * @param {Object} req The express request.\n   * @returns {String|undefined} The locale, or undefined if no locale was set.\n   */\n  getLocale(req) {\n    const locale =\n      (req.query || {})[pageParams.locale] ||\n      (req.body || {})[pageParams.locale] ||\n      (req.params || {})[pageParams.locale] ||\n      (req.headers || {})[pageParamHeaderPrefix + pageParams.locale];\n    return locale;\n  }\n\n  /**\n   * Creates a response with http redirect.\n   * @param {Object} req The express request.\n   * @param {String} path The path of the file to return.\n   * @param {Object} params The query parameters to include.\n   * @returns {Object} The Promise Router response.\n   */\n  async redirectResponse(url, params) {\n    // Remove any parameters with undefined value\n    params = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[p[0]] = p[1];\n      }\n      return m;\n    }, {});\n\n    // Compose URL with parameters in query\n    const location = new URL(url);\n    Object.entries(params).forEach(p => location.searchParams.set(p[0], p[1]));\n    const locationString = location.toString();\n\n    // Add parameters to header to allow parsing for programmatic use\n    // of response, instead of having to parse the HTML content.\n    const headers = Object.entries(params).reduce((m, p) => {\n      if (p[1] !== undefined) {\n        m[`${pageParamHeaderPrefix}${p[0].toLowerCase()}`] = p[1];\n      }\n      return m;\n    }, {});\n\n    return {\n      status: 303,\n      location: locationString,\n      headers: headers,\n    };\n  }\n\n  defaultPagePath(file) {\n    return path.join(this.pagesPath, file);\n  }\n\n  composePageUrl(file, publicServerUrl, locale) {\n    let url = publicServerUrl;\n    url += url.endsWith('/') ? '' : '/';\n    url += this.pagesEndpoint + '/';\n    url += locale === undefined ? '' : locale + '/';\n    url += file;\n    return url;\n  }\n\n  notFound() {\n    return {\n      text: 'Not found.',\n      status: 404,\n    };\n  }\n\n  invalidRequest() {\n    const error = new Error();\n    error.status = 403;\n    error.message = 'unauthorized';\n    throw error;\n  }\n\n  /**\n   * Sets the Parse Server configuration in the request object to make it\n   * easily accessible throughtout request processing.\n   * @param {Object} req The request.\n   * @param {Boolean} failGracefully Is true if failing to set the config should\n   * not result in an invalid request response. Default is `false`.\n   */\n  setConfig(req, failGracefully = false) {\n    req.config = Config.get(req.params.appId || req.query.appId);\n    if (!req.config && !failGracefully) {\n      this.invalidRequest();\n    }\n    return Promise.resolve();\n  }\n\n  mountPagesRoutes() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/verify_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.verifyEmail(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/resend_verification_email`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resendVerificationEmail(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/choose_password`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.passwordReset(req);\n      }\n    );\n\n    this.route(\n      'POST',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.resetPassword(req);\n      }\n    );\n\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/:appId/request_password_reset`,\n      req => {\n        this.setConfig(req);\n      },\n      req => {\n        return this.requestResetPassword(req);\n      }\n    );\n  }\n\n  mountCustomRoutes() {\n    for (const route of this.pagesConfig.customRoutes || []) {\n      this.route(\n        route.method,\n        `/${this.pagesEndpoint}/:appId/${route.path}`,\n        req => {\n          this.setConfig(req);\n        },\n        async req => {\n          const { file, query = {} } = (await route.handler(req)) || {};\n\n          // If route handler did not return a page send 404 response\n          if (!file) {\n            return this.notFound();\n          }\n\n          // Send page response\n          const page = new Page({ id: file, defaultFile: file });\n          return this.goToPage(req, page, query, false);\n        }\n      );\n    }\n  }\n\n  mountStaticRoute() {\n    this.route(\n      'GET',\n      `/${this.pagesEndpoint}/(*)?`,\n      req => {\n        this.setConfig(req, true);\n      },\n      req => {\n        return this.staticRoute(req);\n      }\n    );\n  }\n\n  expressRouter() {\n    const router = express.Router();\n    router.use('/', super.expressRouter());\n    return router;\n  }\n}\n\nexport default PagesRouter;\nmodule.exports = {\n  PagesRouter,\n  pageParamHeaderPrefix,\n  pageParams,\n  pages,\n};\n"],"mappings":";;;;;;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAA2B;AAE3B;AACA,MAAMA,KAAK,GAAGC,MAAM,CAACC,MAAM,CAAC;EAC1BC,aAAa,EAAE,IAAIC,aAAI,CAAC;IAAEC,EAAE,EAAE,eAAe;IAAEC,WAAW,EAAE;EAAsB,CAAC,CAAC;EACpFC,oBAAoB,EAAE,IAAIH,aAAI,CAAC;IAC7BC,EAAE,EAAE,sBAAsB;IAC1BC,WAAW,EAAE;EACf,CAAC,CAAC;EACFE,wBAAwB,EAAE,IAAIJ,aAAI,CAAC;IACjCC,EAAE,EAAE,0BAA0B;IAC9BC,WAAW,EAAE;EACf,CAAC,CAAC;EACFG,wBAAwB,EAAE,IAAIL,aAAI,CAAC;IACjCC,EAAE,EAAE,0BAA0B;IAC9BC,WAAW,EAAE;EACf,CAAC,CAAC;EACFI,yBAAyB,EAAE,IAAIN,aAAI,CAAC;IAClCC,EAAE,EAAE,2BAA2B;IAC/BC,WAAW,EAAE;EACf,CAAC,CAAC;EACFK,4BAA4B,EAAE,IAAIP,aAAI,CAAC;IACrCC,EAAE,EAAE,8BAA8B;IAClCC,WAAW,EAAE;EACf,CAAC,CAAC;EACFM,4BAA4B,EAAE,IAAIR,aAAI,CAAC;IACrCC,EAAE,EAAE,8BAA8B;IAClCC,WAAW,EAAE;EACf,CAAC,CAAC;EACFO,4BAA4B,EAAE,IAAIT,aAAI,CAAC;IACrCC,EAAE,EAAE,8BAA8B;IAClCC,WAAW,EAAE;EACf,CAAC;AACH,CAAC,CAAC;;AAEF;AACA,MAAMQ,UAAU,GAAGb,MAAM,CAACC,MAAM,CAAC;EAC/Ba,OAAO,EAAE,SAAS;EAClBC,KAAK,EAAE,OAAO;EACdC,KAAK,EAAE,OAAO;EACdC,QAAQ,EAAE,UAAU;EACpBC,KAAK,EAAE,OAAO;EACdC,MAAM,EAAE,QAAQ;EAChBC,eAAe,EAAE;AACnB,CAAC,CAAC;;AAEF;AACA,MAAMC,qBAAqB,GAAG,qBAAqB;;AAEnD;AACA,MAAMC,MAAM,GAAGtB,MAAM,CAACC,MAAM,CAAC;EAC3BsB,qBAAqB,EAAE,0BAA0B;EACjDC,uBAAuB,EAAE;AAC3B,CAAC,CAAC;AAEK,MAAMC,WAAW,SAASC,sBAAa,CAAC;EAC7C;AACF;AACA;AACA;EACEC,WAAW,CAAC5B,KAAK,GAAG,CAAC,CAAC,EAAE;IACtB,KAAK,EAAE;;IAEP;IACA,IAAI,CAAC6B,WAAW,GAAG7B,KAAK;IACxB,IAAI,CAAC8B,aAAa,GAAG9B,KAAK,CAAC8B,aAAa,GAAG9B,KAAK,CAAC8B,aAAa,GAAG,MAAM;IACvE,IAAI,CAACC,SAAS,GAAG/B,KAAK,CAAC+B,SAAS,GAC5BC,aAAI,CAACC,OAAO,CAAC,IAAI,EAAEjC,KAAK,CAAC+B,SAAS,CAAC,GACnCC,aAAI,CAACC,OAAO,CAACC,SAAS,EAAE,cAAc,CAAC;IAC3C,IAAI,CAACC,gBAAgB,EAAE;IACvB,IAAI,CAACC,gBAAgB,EAAE;IACvB,IAAI,CAACC,iBAAiB,EAAE;IACxB,IAAI,CAACC,gBAAgB,EAAE;EACzB;EAEAC,WAAW,CAACC,GAAG,EAAE;IACf,MAAMC,MAAM,GAAGD,GAAG,CAACC,MAAM;IACzB,MAAM;MAAEvB,QAAQ;MAAED,KAAK,EAAEyB;IAAS,CAAC,GAAGF,GAAG,CAACG,KAAK;IAC/C,MAAM1B,KAAK,GAAGyB,QAAQ,IAAI,OAAOA,QAAQ,KAAK,QAAQ,GAAGA,QAAQ,CAACE,QAAQ,EAAE,GAAGF,QAAQ;IAEvF,IAAI,CAACD,MAAM,EAAE;MACX,IAAI,CAACI,cAAc,EAAE;IACvB;IAEA,IAAI,CAAC5B,KAAK,IAAI,CAACC,QAAQ,EAAE;MACvB,OAAO,IAAI,CAAC4B,QAAQ,CAACN,GAAG,EAAExC,KAAK,CAACY,4BAA4B,CAAC;IAC/D;IAEA,MAAMmC,cAAc,GAAGN,MAAM,CAACM,cAAc;IAC5C,OAAOA,cAAc,CAACR,WAAW,CAACrB,QAAQ,EAAED,KAAK,CAAC,CAAC+B,IAAI,CACrD,MAAM;MACJ,MAAMC,MAAM,GAAG;QACb,CAACnC,UAAU,CAACI,QAAQ,GAAGA;MACzB,CAAC;MACD,OAAO,IAAI,CAAC4B,QAAQ,CAACN,GAAG,EAAExC,KAAK,CAACS,wBAAwB,EAAEwC,MAAM,CAAC;IACnE,CAAC,EACD,MAAM;MACJ,MAAMA,MAAM,GAAG;QACb,CAACnC,UAAU,CAACI,QAAQ,GAAGA;MACzB,CAAC;MACD,OAAO,IAAI,CAAC4B,QAAQ,CAACN,GAAG,EAAExC,KAAK,CAACa,4BAA4B,EAAEoC,MAAM,CAAC;IACvE,CAAC,CACF;EACH;EAEAC,uBAAuB,CAACV,GAAG,EAAE;IAC3B,MAAMC,MAAM,GAAGD,GAAG,CAACC,MAAM;IACzB,MAAMvB,QAAQ,GAAGsB,GAAG,CAACW,IAAI,CAACjC,QAAQ;IAElC,IAAI,CAACuB,MAAM,EAAE;MACX,IAAI,CAACI,cAAc,EAAE;IACvB;IAEA,IAAI,CAAC3B,QAAQ,EAAE;MACb,OAAO,IAAI,CAAC4B,QAAQ,CAACN,GAAG,EAAExC,KAAK,CAACY,4BAA4B,CAAC;IAC/D;IAEA,MAAMmC,cAAc,GAAGN,MAAM,CAACM,cAAc;IAE5C,OAAOA,cAAc,CAACG,uBAAuB,CAAChC,QAAQ,CAAC,CAAC8B,IAAI,CAC1D,MAAM;MACJ,OAAO,IAAI,CAACF,QAAQ,CAACN,GAAG,EAAExC,KAAK,CAACW,4BAA4B,CAAC;IAC/D,CAAC,EACD,MAAM;MACJ,OAAO,IAAI,CAACmC,QAAQ,CAACN,GAAG,EAAExC,KAAK,CAACU,yBAAyB,CAAC;IAC5D,CAAC,CACF;EACH;EAEAP,aAAa,CAACqC,GAAG,EAAE;IACjB,MAAMC,MAAM,GAAGD,GAAG,CAACC,MAAM;IACzB,MAAMQ,MAAM,GAAG;MACb,CAACnC,UAAU,CAACE,KAAK,GAAGwB,GAAG,CAACS,MAAM,CAACjC,KAAK;MACpC,CAACF,UAAU,CAACC,OAAO,GAAG0B,MAAM,CAAC1B,OAAO;MACpC,CAACD,UAAU,CAACG,KAAK,GAAGuB,GAAG,CAACG,KAAK,CAAC1B,KAAK;MACnC,CAACH,UAAU,CAACI,QAAQ,GAAGsB,GAAG,CAACG,KAAK,CAACzB,QAAQ;MACzC,CAACJ,UAAU,CAACO,eAAe,GAAGoB,MAAM,CAACW;IACvC,CAAC;IACD,OAAO,IAAI,CAACN,QAAQ,CAACN,GAAG,EAAExC,KAAK,CAACG,aAAa,EAAE8C,MAAM,CAAC;EACxD;EAEAI,oBAAoB,CAACb,GAAG,EAAE;IACxB,MAAMC,MAAM,GAAGD,GAAG,CAACC,MAAM;IAEzB,IAAI,CAACA,MAAM,EAAE;MACX,IAAI,CAACI,cAAc,EAAE;IACvB;IAEA,MAAM;MAAE3B,QAAQ;MAAED,KAAK,EAAEyB;IAAS,CAAC,GAAGF,GAAG,CAACG,KAAK;IAC/C,MAAM1B,KAAK,GAAGyB,QAAQ,IAAI,OAAOA,QAAQ,KAAK,QAAQ,GAAGA,QAAQ,CAACE,QAAQ,EAAE,GAAGF,QAAQ;IAEvF,IAAI,CAACxB,QAAQ,IAAI,CAACD,KAAK,EAAE;MACvB,OAAO,IAAI,CAAC6B,QAAQ,CAACN,GAAG,EAAExC,KAAK,CAACQ,wBAAwB,CAAC;IAC3D;IAEA,OAAOiC,MAAM,CAACM,cAAc,CAACO,uBAAuB,CAACpC,QAAQ,EAAED,KAAK,CAAC,CAAC+B,IAAI,CACxE,MAAM;MACJ,MAAMC,MAAM,GAAG;QACb,CAACnC,UAAU,CAACG,KAAK,GAAGA,KAAK;QACzB,CAACH,UAAU,CAACI,QAAQ,GAAGA,QAAQ;QAC/B,CAACJ,UAAU,CAACE,KAAK,GAAGyB,MAAM,CAACc,aAAa;QACxC,CAACzC,UAAU,CAACC,OAAO,GAAG0B,MAAM,CAAC1B;MAC/B,CAAC;MACD,OAAO,IAAI,CAAC+B,QAAQ,CAACN,GAAG,EAAExC,KAAK,CAACG,aAAa,EAAE8C,MAAM,CAAC;IACxD,CAAC,EACD,MAAM;MACJ,MAAMA,MAAM,GAAG;QACb,CAACnC,UAAU,CAACI,QAAQ,GAAGA;MACzB,CAAC;MACD,OAAO,IAAI,CAAC4B,QAAQ,CAACN,GAAG,EAAExC,KAAK,CAACQ,wBAAwB,EAAEyC,MAAM,CAAC;IACnE,CAAC,CACF;EACH;EAEAO,aAAa,CAAChB,GAAG,EAAE;IACjB,MAAMC,MAAM,GAAGD,GAAG,CAACC,MAAM;IAEzB,IAAI,CAACA,MAAM,EAAE;MACX,IAAI,CAACI,cAAc,EAAE;IACvB;IAEA,MAAM;MAAE3B,QAAQ;MAAEuC,YAAY;MAAExC,KAAK,EAAEyB;IAAS,CAAC,GAAGF,GAAG,CAACW,IAAI;IAC5D,MAAMlC,KAAK,GAAGyB,QAAQ,IAAI,OAAOA,QAAQ,KAAK,QAAQ,GAAGA,QAAQ,CAACE,QAAQ,EAAE,GAAGF,QAAQ;IAEvF,IAAI,CAAC,CAACxB,QAAQ,IAAI,CAACD,KAAK,IAAI,CAACwC,YAAY,KAAKjB,GAAG,CAACkB,GAAG,KAAK,KAAK,EAAE;MAC/D,OAAO,IAAI,CAACZ,QAAQ,CAACN,GAAG,EAAExC,KAAK,CAACQ,wBAAwB,CAAC;IAC3D;IAEA,IAAI,CAACU,QAAQ,EAAE;MACb,MAAM,IAAIyC,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACC,gBAAgB,EAAE,kBAAkB,CAAC;IACzE;IAEA,IAAI,CAAC5C,KAAK,EAAE;MACV,MAAM,IAAI0C,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACE,WAAW,EAAE,eAAe,CAAC;IACjE;IAEA,IAAI,CAACL,YAAY,EAAE;MACjB,MAAM,IAAIE,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACG,gBAAgB,EAAE,kBAAkB,CAAC;IACzE;IAEA,OAAOtB,MAAM,CAACM,cAAc,CACzBiB,cAAc,CAAC9C,QAAQ,EAAED,KAAK,EAAEwC,YAAY,CAAC,CAC7CT,IAAI,CACH,MAAM;MACJ,OAAOiB,OAAO,CAAChC,OAAO,CAAC;QACrBiC,OAAO,EAAE;MACX,CAAC,CAAC;IACJ,CAAC,EACDC,GAAG,IAAI;MACL,OAAOF,OAAO,CAAChC,OAAO,CAAC;QACrBiC,OAAO,EAAE,KAAK;QACdC;MACF,CAAC,CAAC;IACJ,CAAC,CACF,CACAnB,IAAI,CAACoB,MAAM,IAAI;MACd,IAAI5B,GAAG,CAACkB,GAAG,EAAE;QACX,IAAIU,MAAM,CAACF,OAAO,EAAE;UAClB,OAAOD,OAAO,CAAChC,OAAO,CAAC;YACrBoC,MAAM,EAAE,GAAG;YACXC,QAAQ,EAAE;UACZ,CAAC,CAAC;QACJ;QACA,IAAIF,MAAM,CAACD,GAAG,EAAE;UACd,MAAM,IAAIR,WAAK,CAACC,KAAK,CAACD,WAAK,CAACC,KAAK,CAACE,WAAW,EAAG,GAAEM,MAAM,CAACD,GAAI,EAAC,CAAC;QACjE;MACF;MAEA,MAAMxB,KAAK,GAAGyB,MAAM,CAACF,OAAO,GACxB;QACA,CAACpD,UAAU,CAACI,QAAQ,GAAGA;MACzB,CAAC,GACC;QACA,CAACJ,UAAU,CAACI,QAAQ,GAAGA,QAAQ;QAC/B,CAACJ,UAAU,CAACG,KAAK,GAAGA,KAAK;QACzB,CAACH,UAAU,CAACE,KAAK,GAAGyB,MAAM,CAACc,aAAa;QACxC,CAACzC,UAAU,CAACK,KAAK,GAAGiD,MAAM,CAACD,GAAG;QAC9B,CAACrD,UAAU,CAACC,OAAO,GAAG0B,MAAM,CAAC1B;MAC/B,CAAC;MACH,MAAMwD,IAAI,GAAGH,MAAM,CAACF,OAAO,GAAGlE,KAAK,CAACO,oBAAoB,GAAGP,KAAK,CAACG,aAAa;MAE9E,OAAO,IAAI,CAAC2C,QAAQ,CAACN,GAAG,EAAE+B,IAAI,EAAE5B,KAAK,EAAE,KAAK,CAAC;IAC/C,CAAC,CAAC;EACN;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACEG,QAAQ,CAACN,GAAG,EAAE+B,IAAI,EAAEtB,MAAM,GAAG,CAAC,CAAC,EAAEuB,YAAY,EAAE;IAC7C,MAAM/B,MAAM,GAAGD,GAAG,CAACC,MAAM;;IAEzB;IACA,MAAMgC,QAAQ,GAAGhC,MAAM,CAACzC,KAAK,CAAC0E,aAAa,GACvC,IAAI,GACJF,YAAY,KAAKG,SAAS,GACxBH,YAAY,GACZhC,GAAG,CAACoC,MAAM,IAAI,MAAM;;IAE1B;IACA,MAAMC,aAAa,GAAG,IAAI,CAACC,gBAAgB,CAACrC,MAAM,CAAC;IACnD,IAAIxC,MAAM,CAAC8E,MAAM,CAACF,aAAa,CAAC,CAACG,QAAQ,CAACL,SAAS,CAAC,EAAE;MACpD,OAAO,IAAI,CAACM,QAAQ,EAAE;IACxB;IACAhC,MAAM,GAAGhD,MAAM,CAACiF,MAAM,CAACjC,MAAM,EAAE4B,aAAa,CAAC;;IAE7C;IACA;IACA;IACA,MAAMzD,MAAM,GAAG,IAAI,CAAC+D,SAAS,CAAC3C,GAAG,CAAC;IAClCS,MAAM,CAACnC,UAAU,CAACM,MAAM,CAAC,GAAGA,MAAM;;IAElC;IACA,MAAMd,WAAW,GAAGiE,IAAI,CAACjE,WAAW;IACpC,MAAM8E,WAAW,GAAG,IAAI,CAACC,eAAe,CAAC/E,WAAW,CAAC;IACrD,MAAMgF,UAAU,GAAG,IAAI,CAACC,cAAc,CAACjF,WAAW,EAAEmC,MAAM,CAACW,eAAe,CAAC;;IAE3E;IACA,MAAMoC,SAAS,GAAG/C,MAAM,CAACzC,KAAK,CAACyF,UAAU,CAAClB,IAAI,CAAClE,EAAE,CAAC;IAClD,IAAImF,SAAS,IAAI,CAACE,cAAK,CAACC,MAAM,CAACH,SAAS,CAAC,EAAE;MACzC,OAAO,IAAI,CAACI,gBAAgB,CAACJ,SAAS,EAAEvC,MAAM,CAAC;IACjD;;IAEA;IACA,IAAI4C,YAAY,GAAG,CAAC,CAAC;IACrB,IAAIpD,MAAM,CAACzC,KAAK,CAAC8F,kBAAkB,IAAIrD,MAAM,CAACzC,KAAK,CAAC+F,oBAAoB,EAAE;MACxEF,YAAY,GAAG,IAAI,CAACG,mBAAmB,CAAC5E,MAAM,EAAE6B,MAAM,CAAC;IACzD;;IAEA;IACA,IAAIR,MAAM,CAACzC,KAAK,CAAC8F,kBAAkB,IAAI1E,MAAM,EAAE;MAC7C,OAAOsE,cAAK,CAACO,gBAAgB,CAACb,WAAW,EAAEhE,MAAM,CAAC,CAAC4B,IAAI,CAAC,CAAC;QAAEhB,IAAI;QAAEkE;MAAO,CAAC,KACvEzB,QAAQ,GACJ,IAAI,CAACmB,gBAAgB,CACrB,IAAI,CAACL,cAAc,CAACjF,WAAW,EAAEmC,MAAM,CAACW,eAAe,EAAE8C,MAAM,CAAC,EAChEjD,MAAM,CACP,GACC,IAAI,CAACkD,YAAY,CAACnE,IAAI,EAAEiB,MAAM,EAAE4C,YAAY,CAAC,CAClD;IACH,CAAC,MAAM;MACL,OAAOpB,QAAQ,GACX,IAAI,CAACmB,gBAAgB,CAACN,UAAU,EAAErC,MAAM,CAAC,GACzC,IAAI,CAACkD,YAAY,CAACf,WAAW,EAAEnC,MAAM,EAAE4C,YAAY,CAAC;IAC1D;EACF;;EAEA;AACF;AACA;AACA;AACA;AACA;EACEO,WAAW,CAAC5D,GAAG,EAAE;IACf;IACA,MAAM6D,YAAY,GAAG7D,GAAG,CAACS,MAAM,CAAC,CAAC,CAAC;;IAElC;IACA,MAAMqD,YAAY,GAAGtE,aAAI,CAACC,OAAO,CAAC,IAAI,CAACF,SAAS,EAAEsE,YAAY,CAAC;;IAE/D;IACA,IAAI,CAACC,YAAY,IAAI,CAACA,YAAY,CAACC,QAAQ,CAAC,OAAO,CAAC,EAAE;MACpD,OAAO,IAAI,CAACC,YAAY,CAACF,YAAY,CAAC;IACxC;;IAEA;IACA,MAAMrD,MAAM,GAAG,IAAI,CAAC6B,gBAAgB,CAACtC,GAAG,CAACC,MAAM,CAAC;IAChD,MAAMrB,MAAM,GAAG,IAAI,CAAC+D,SAAS,CAAC3C,GAAG,CAAC;IAClC,IAAIpB,MAAM,EAAE;MACV6B,MAAM,CAAC7B,MAAM,GAAGA,MAAM;IACxB;;IAEA;IACA,MAAMyE,YAAY,GAAG,IAAI,CAACG,mBAAmB,CAAC5E,MAAM,EAAE6B,MAAM,CAAC;IAE7D,OAAO,IAAI,CAACkD,YAAY,CAACG,YAAY,EAAErD,MAAM,EAAE4C,YAAY,CAAC;EAC9D;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACEY,kBAAkB,CAACrF,MAAM,EAAE;IACzB;IACA,IAAI,IAAI,CAACsF,cAAc,KAAK/B,SAAS,EAAE;MACrC,OAAO,CAAC,CAAC;IACX;;IAEA;IACAvD,MAAM,GAAGA,MAAM,IAAI,IAAI,CAACS,WAAW,CAAC8E,0BAA0B;;IAE9D;IACA,MAAMC,QAAQ,GAAGxF,MAAM,CAACyF,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;IACrC,MAAMC,QAAQ,GACZ,IAAI,CAACJ,cAAc,CAACtF,MAAM,CAAC,IAC3B,IAAI,CAACsF,cAAc,CAACE,QAAQ,CAAC,IAC7B,IAAI,CAACF,cAAc,CAAC,IAAI,CAAC7E,WAAW,CAAC8E,0BAA0B,CAAC,IAChE,CAAC,CAAC;IACJ,MAAMI,WAAW,GAAGD,QAAQ,CAACC,WAAW,IAAI,CAAC,CAAC;IAC9C,OAAOA,WAAW;EACpB;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACEf,mBAAmB,CAAC5E,MAAM,EAAE6B,MAAM,GAAG,CAAC,CAAC,EAAE;IACvC;IACA,IAAI,CAAC,IAAI,CAACpB,WAAW,CAACiE,kBAAkB,IAAI,CAAC,IAAI,CAACjE,WAAW,CAACkE,oBAAoB,EAAE;MAClF,OAAO,CAAC,CAAC;IACX;;IAEA;IACA,IAAIF,YAAY,GAAG,IAAI,CAACY,kBAAkB,CAACrF,MAAM,CAAC;;IAElD;IACA;IACAyE,YAAY,GAAGmB,IAAI,CAACC,SAAS,CAACpB,YAAY,CAAC;IAC3CA,YAAY,GAAGqB,iBAAQ,CAACC,MAAM,CAACtB,YAAY,EAAE5C,MAAM,CAAC;IACpD4C,YAAY,GAAGmB,IAAI,CAACI,KAAK,CAACvB,YAAY,CAAC;IAEvC,OAAOA,YAAY;EACrB;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACE,MAAMM,YAAY,CAACnE,IAAI,EAAEiB,MAAM,GAAG,CAAC,CAAC,EAAE4C,YAAY,GAAG,CAAC,CAAC,EAAE;IACvD;IACA,IAAIwB,IAAI;IACR,IAAI;MACFA,IAAI,GAAG,MAAM,IAAI,CAACC,QAAQ,CAACtF,IAAI,CAAC;IAClC,CAAC,CAAC,OAAOuF,CAAC,EAAE;MACV,OAAO,IAAI,CAACtC,QAAQ,EAAE;IACxB;;IAEA;IACA,IAAIuC,kBAAkB,GACpB,OAAO,IAAI,CAAC3F,WAAW,CAACgE,YAAY,KAAK,UAAU,GAC/C,IAAI,CAAChE,WAAW,CAACgE,YAAY,CAAC5C,MAAM,CAAC,GACrChD,MAAM,CAACwH,SAAS,CAAC7E,QAAQ,CAAC8E,IAAI,CAAC,IAAI,CAAC7F,WAAW,CAACgE,YAAY,CAAC,KAAK,iBAAiB,GACjF,IAAI,CAAChE,WAAW,CAACgE,YAAY,GAC7B,CAAC,CAAC;IACV,IAAI2B,kBAAkB,YAAYvD,OAAO,EAAE;MACzCuD,kBAAkB,GAAG,MAAMA,kBAAkB;IAC/C;;IAEA;IACA,MAAMG,eAAe,GAAG1H,MAAM,CAACiF,MAAM,CAAC,CAAC,CAAC,EAAEsC,kBAAkB,EAAE3B,YAAY,CAAC;IAC3E,MAAM+B,qBAAqB,GAAG3H,MAAM,CAACiF,MAAM,CAAC,CAAC,CAAC,EAAEjC,MAAM,EAAE0E,eAAe,CAAC;IACxEN,IAAI,GAAGH,iBAAQ,CAACC,MAAM,CAACE,IAAI,EAAEO,qBAAqB,CAAC;;IAEnD;IACA;IACA,MAAMC,OAAO,GAAG5H,MAAM,CAAC6H,OAAO,CAAC7E,MAAM,CAAC,CAAC8E,MAAM,CAAC,CAACC,CAAC,EAAEC,CAAC,KAAK;MACtD,IAAIA,CAAC,CAAC,CAAC,CAAC,KAAKtD,SAAS,EAAE;QACtBqD,CAAC,CAAE,GAAE1G,qBAAsB,GAAE2G,CAAC,CAAC,CAAC,CAAC,CAACC,WAAW,EAAG,EAAC,CAAC,GAAGD,CAAC,CAAC,CAAC,CAAC;MAC3D;MACA,OAAOD,CAAC;IACV,CAAC,EAAE,CAAC,CAAC,CAAC;IAEN,OAAO;MAAEG,IAAI,EAAEd,IAAI;MAAEQ,OAAO,EAAEA;IAAQ,CAAC;EACzC;;EAEA;AACF;AACA;AACA;AACA;EACE,MAAMrB,YAAY,CAACxE,IAAI,EAAE;IACvB;IACA,IAAIqF,IAAI;IACR,IAAI;MACFA,IAAI,GAAG,MAAM,IAAI,CAACC,QAAQ,CAACtF,IAAI,CAAC;IAClC,CAAC,CAAC,OAAOuF,CAAC,EAAE;MACV,OAAO,IAAI,CAACtC,QAAQ,EAAE;IACxB;IAEA,OAAO;MAAEkD,IAAI,EAAEd;IAAK,CAAC;EACvB;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;EACE,MAAMC,QAAQ,CAACc,QAAQ,EAAE;IACvB;IACA;IACA;IACA;IACA,MAAMC,cAAc,GAAGrG,aAAI,CAACsG,SAAS,CAACF,QAAQ,CAAC;;IAE/C;IACA,IAAI,CAACC,cAAc,CAACE,UAAU,CAAC,IAAI,CAACxG,SAAS,CAAC,EAAE;MAC9C,MAAMR,MAAM,CAACE,uBAAuB;IACtC;IAEA,OAAO,MAAM+G,YAAE,CAAClB,QAAQ,CAACe,cAAc,EAAE,OAAO,CAAC;EACnD;;EAEA;AACF;AACA;EACElG,gBAAgB,GAAG;IACjB,IAAI,IAAI,CAACN,WAAW,CAACkE,oBAAoB,KAAKpB,SAAS,EAAE;MACvD;IACF;IACA,IAAI;MACF,MAAM8D,IAAI,GAAGC,OAAO,CAAC1G,aAAI,CAACC,OAAO,CAAC,IAAI,EAAE,IAAI,CAACJ,WAAW,CAACkE,oBAAoB,CAAC,CAAC;MAC/E,IAAI,CAACW,cAAc,GAAG+B,IAAI;IAC5B,CAAC,CAAC,OAAOlB,CAAC,EAAE;MACV,MAAMhG,MAAM,CAACC,qBAAqB;IACpC;EACF;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;EACEsD,gBAAgB,CAACrC,MAAM,EAAE;IACvB,OAAOA,MAAM,GACT;MACA,CAAC3B,UAAU,CAACE,KAAK,GAAGyB,MAAM,CAACzB,KAAK;MAChC,CAACF,UAAU,CAACC,OAAO,GAAG0B,MAAM,CAAC1B,OAAO;MACpC,CAACD,UAAU,CAACO,eAAe,GAAGoB,MAAM,CAACW;IACvC,CAAC,GACC,CAAC,CAAC;EACR;;EAEA;AACF;AACA;AACA;AACA;EACE+B,SAAS,CAAC3C,GAAG,EAAE;IACb,MAAMpB,MAAM,GACV,CAACoB,GAAG,CAACG,KAAK,IAAI,CAAC,CAAC,EAAE7B,UAAU,CAACM,MAAM,CAAC,IACpC,CAACoB,GAAG,CAACW,IAAI,IAAI,CAAC,CAAC,EAAErC,UAAU,CAACM,MAAM,CAAC,IACnC,CAACoB,GAAG,CAACS,MAAM,IAAI,CAAC,CAAC,EAAEnC,UAAU,CAACM,MAAM,CAAC,IACrC,CAACoB,GAAG,CAACqF,OAAO,IAAI,CAAC,CAAC,EAAEvG,qBAAqB,GAAGR,UAAU,CAACM,MAAM,CAAC;IAChE,OAAOA,MAAM;EACf;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;EACE,MAAMwE,gBAAgB,CAAC+C,GAAG,EAAE1F,MAAM,EAAE;IAClC;IACAA,MAAM,GAAGhD,MAAM,CAAC6H,OAAO,CAAC7E,MAAM,CAAC,CAAC8E,MAAM,CAAC,CAACC,CAAC,EAAEC,CAAC,KAAK;MAC/C,IAAIA,CAAC,CAAC,CAAC,CAAC,KAAKtD,SAAS,EAAE;QACtBqD,CAAC,CAACC,CAAC,CAAC,CAAC,CAAC,CAAC,GAAGA,CAAC,CAAC,CAAC,CAAC;MAChB;MACA,OAAOD,CAAC;IACV,CAAC,EAAE,CAAC,CAAC,CAAC;;IAEN;IACA,MAAMY,QAAQ,GAAG,IAAIC,GAAG,CAACF,GAAG,CAAC;IAC7B1I,MAAM,CAAC6H,OAAO,CAAC7E,MAAM,CAAC,CAAC6F,OAAO,CAACb,CAAC,IAAIW,QAAQ,CAACG,YAAY,CAACC,GAAG,CAACf,CAAC,CAAC,CAAC,CAAC,EAAEA,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAC1E,MAAMgB,cAAc,GAAGL,QAAQ,CAAChG,QAAQ,EAAE;;IAE1C;IACA;IACA,MAAMiF,OAAO,GAAG5H,MAAM,CAAC6H,OAAO,CAAC7E,MAAM,CAAC,CAAC8E,MAAM,CAAC,CAACC,CAAC,EAAEC,CAAC,KAAK;MACtD,IAAIA,CAAC,CAAC,CAAC,CAAC,KAAKtD,SAAS,EAAE;QACtBqD,CAAC,CAAE,GAAE1G,qBAAsB,GAAE2G,CAAC,CAAC,CAAC,CAAC,CAACC,WAAW,EAAG,EAAC,CAAC,GAAGD,CAAC,CAAC,CAAC,CAAC;MAC3D;MACA,OAAOD,CAAC;IACV,CAAC,EAAE,CAAC,CAAC,CAAC;IAEN,OAAO;MACL3D,MAAM,EAAE,GAAG;MACXuE,QAAQ,EAAEK,cAAc;MACxBpB,OAAO,EAAEA;IACX,CAAC;EACH;EAEAxC,eAAe,CAAC6D,IAAI,EAAE;IACpB,OAAOlH,aAAI,CAACmH,IAAI,CAAC,IAAI,CAACpH,SAAS,EAAEmH,IAAI,CAAC;EACxC;EAEA3D,cAAc,CAAC2D,IAAI,EAAE7H,eAAe,EAAED,MAAM,EAAE;IAC5C,IAAIuH,GAAG,GAAGtH,eAAe;IACzBsH,GAAG,IAAIA,GAAG,CAACpC,QAAQ,CAAC,GAAG,CAAC,GAAG,EAAE,GAAG,GAAG;IACnCoC,GAAG,IAAI,IAAI,CAAC7G,aAAa,GAAG,GAAG;IAC/B6G,GAAG,IAAIvH,MAAM,KAAKuD,SAAS,GAAG,EAAE,GAAGvD,MAAM,GAAG,GAAG;IAC/CuH,GAAG,IAAIO,IAAI;IACX,OAAOP,GAAG;EACZ;EAEA1D,QAAQ,GAAG;IACT,OAAO;MACLkD,IAAI,EAAE,YAAY;MAClB9D,MAAM,EAAE;IACV,CAAC;EACH;EAEAxB,cAAc,GAAG;IACf,MAAM1B,KAAK,GAAG,IAAIyC,KAAK,EAAE;IACzBzC,KAAK,CAACkD,MAAM,GAAG,GAAG;IAClBlD,KAAK,CAACiI,OAAO,GAAG,cAAc;IAC9B,MAAMjI,KAAK;EACb;;EAEA;AACF;AACA;AACA;AACA;AACA;AACA;EACEkI,SAAS,CAAC7G,GAAG,EAAE8G,cAAc,GAAG,KAAK,EAAE;IACrC9G,GAAG,CAACC,MAAM,GAAG8G,eAAM,CAACC,GAAG,CAAChH,GAAG,CAACS,MAAM,CAACjC,KAAK,IAAIwB,GAAG,CAACG,KAAK,CAAC3B,KAAK,CAAC;IAC5D,IAAI,CAACwB,GAAG,CAACC,MAAM,IAAI,CAAC6G,cAAc,EAAE;MAClC,IAAI,CAACzG,cAAc,EAAE;IACvB;IACA,OAAOoB,OAAO,CAAChC,OAAO,EAAE;EAC1B;EAEAG,gBAAgB,GAAG;IACjB,IAAI,CAACqH,KAAK,CACR,KAAK,EACJ,IAAG,IAAI,CAAC3H,aAAc,sBAAqB,EAC5CU,GAAG,IAAI;MACL,IAAI,CAAC6G,SAAS,CAAC7G,GAAG,CAAC;IACrB,CAAC,EACDA,GAAG,IAAI;MACL,OAAO,IAAI,CAACD,WAAW,CAACC,GAAG,CAAC;IAC9B,CAAC,CACF;IAED,IAAI,CAACiH,KAAK,CACR,MAAM,EACL,IAAG,IAAI,CAAC3H,aAAc,mCAAkC,EACzDU,GAAG,IAAI;MACL,IAAI,CAAC6G,SAAS,CAAC7G,GAAG,CAAC;IACrB,CAAC,EACDA,GAAG,IAAI;MACL,OAAO,IAAI,CAACU,uBAAuB,CAACV,GAAG,CAAC;IAC1C,CAAC,CACF;IAED,IAAI,CAACiH,KAAK,CACR,KAAK,EACJ,IAAG,IAAI,CAAC3H,aAAc,kBAAiB,EACxCU,GAAG,IAAI;MACL,IAAI,CAAC6G,SAAS,CAAC7G,GAAG,CAAC;IACrB,CAAC,EACDA,GAAG,IAAI;MACL,OAAO,IAAI,CAACrC,aAAa,CAACqC,GAAG,CAAC;IAChC,CAAC,CACF;IAED,IAAI,CAACiH,KAAK,CACR,MAAM,EACL,IAAG,IAAI,CAAC3H,aAAc,gCAA+B,EACtDU,GAAG,IAAI;MACL,IAAI,CAAC6G,SAAS,CAAC7G,GAAG,CAAC;IACrB,CAAC,EACDA,GAAG,IAAI;MACL,OAAO,IAAI,CAACgB,aAAa,CAAChB,GAAG,CAAC;IAChC,CAAC,CACF;IAED,IAAI,CAACiH,KAAK,CACR,KAAK,EACJ,IAAG,IAAI,CAAC3H,aAAc,gCAA+B,EACtDU,GAAG,IAAI;MACL,IAAI,CAAC6G,SAAS,CAAC7G,GAAG,CAAC;IACrB,CAAC,EACDA,GAAG,IAAI;MACL,OAAO,IAAI,CAACa,oBAAoB,CAACb,GAAG,CAAC;IACvC,CAAC,CACF;EACH;EAEAH,iBAAiB,GAAG;IAClB,KAAK,MAAMoH,KAAK,IAAI,IAAI,CAAC5H,WAAW,CAAC6H,YAAY,IAAI,EAAE,EAAE;MACvD,IAAI,CAACD,KAAK,CACRA,KAAK,CAAC7E,MAAM,EACX,IAAG,IAAI,CAAC9C,aAAc,WAAU2H,KAAK,CAACzH,IAAK,EAAC,EAC7CQ,GAAG,IAAI;QACL,IAAI,CAAC6G,SAAS,CAAC7G,GAAG,CAAC;MACrB,CAAC,EACD,MAAMA,GAAG,IAAI;QACX,MAAM;UAAE0G,IAAI;UAAEvG,KAAK,GAAG,CAAC;QAAE,CAAC,GAAG,CAAC,MAAM8G,KAAK,CAACE,OAAO,CAACnH,GAAG,CAAC,KAAK,CAAC,CAAC;;QAE7D;QACA,IAAI,CAAC0G,IAAI,EAAE;UACT,OAAO,IAAI,CAACjE,QAAQ,EAAE;QACxB;;QAEA;QACA,MAAMV,IAAI,GAAG,IAAInE,aAAI,CAAC;UAAEC,EAAE,EAAE6I,IAAI;UAAE5I,WAAW,EAAE4I;QAAK,CAAC,CAAC;QACtD,OAAO,IAAI,CAACpG,QAAQ,CAACN,GAAG,EAAE+B,IAAI,EAAE5B,KAAK,EAAE,KAAK,CAAC;MAC/C,CAAC,CACF;IACH;EACF;EAEAL,gBAAgB,GAAG;IACjB,IAAI,CAACmH,KAAK,CACR,KAAK,EACJ,IAAG,IAAI,CAAC3H,aAAc,OAAM,EAC7BU,GAAG,IAAI;MACL,IAAI,CAAC6G,SAAS,CAAC7G,GAAG,EAAE,IAAI,CAAC;IAC3B,CAAC,EACDA,GAAG,IAAI;MACL,OAAO,IAAI,CAAC4D,WAAW,CAAC5D,GAAG,CAAC;IAC9B,CAAC,CACF;EACH;EAEAoH,aAAa,GAAG;IACd,MAAMC,MAAM,GAAGC,gBAAO,CAACC,MAAM,EAAE;IAC/BF,MAAM,CAACG,GAAG,CAAC,GAAG,EAAE,KAAK,CAACJ,aAAa,EAAE,CAAC;IACtC,OAAOC,MAAM;EACf;AACF;AAAC;AAAA,eAEcnI,WAAW;AAAA;AAC1BuI,MAAM,CAACC,OAAO,GAAG;EACfxI,WAAW;EACXJ,qBAAqB;EACrBR,UAAU;EACVd;AACF,CAAC"}