@webiny/handler 0.0.0-unstable.990c3ab1b6 → 0.0.0-unstable.bca7b3e350

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.
package/Context.d.ts CHANGED
@@ -9,5 +9,6 @@ export declare class Context extends BaseContext implements BaseContextType {
9
9
  readonly routes: BaseContextType["routes"];
10
10
  handlerClient: BaseContextType["handlerClient"];
11
11
  request: BaseContextType["request"];
12
+ reply: BaseContextType["reply"];
12
13
  constructor(params: ContextParams);
13
14
  }
package/Context.js CHANGED
@@ -1,25 +1,31 @@
1
1
  "use strict";
2
2
 
3
3
  var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault").default;
4
+
4
5
  Object.defineProperty(exports, "__esModule", {
5
6
  value: true
6
7
  });
7
8
  exports.Context = void 0;
9
+
8
10
  var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
11
+
9
12
  var _api = require("@webiny/api");
13
+
10
14
  class Context extends _api.Context {
11
15
  // @ts-ignore
12
-
13
16
  // @ts-ignore
14
-
17
+ // @ts-ignore
15
18
  constructor(params) {
16
19
  super(params);
17
20
  (0, _defineProperty2.default)(this, "server", void 0);
18
21
  (0, _defineProperty2.default)(this, "routes", void 0);
19
22
  (0, _defineProperty2.default)(this, "handlerClient", void 0);
20
23
  (0, _defineProperty2.default)(this, "request", void 0);
24
+ (0, _defineProperty2.default)(this, "reply", void 0);
21
25
  this.server = params.server;
22
26
  this.routes = params.routes;
23
27
  }
28
+
24
29
  }
30
+
25
31
  exports.Context = Context;
package/Context.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"names":["Context","BaseContext","constructor","params","server","routes"],"sources":["Context.ts"],"sourcesContent":["import { Context as BaseContext, ContextParams as BaseContextParams } from \"@webiny/api\";\nimport { Context as BaseContextType } from \"~/types\";\n\nexport interface ContextParams extends BaseContextParams {\n server: BaseContextType[\"server\"];\n routes: BaseContextType[\"routes\"];\n}\n\nexport class Context extends BaseContext implements BaseContextType {\n public readonly server: BaseContextType[\"server\"];\n public readonly routes: BaseContextType[\"routes\"];\n // @ts-ignore\n public handlerClient: BaseContextType[\"handlerClient\"];\n // @ts-ignore\n public request: BaseContextType[\"request\"];\n\n public constructor(params: ContextParams) {\n super(params);\n this.server = params.server;\n this.routes = params.routes;\n }\n}\n"],"mappings":";;;;;;;;AAAA;AAQO,MAAMA,OAAO,SAASC,YAAW,CAA4B;EAGhE;;EAEA;;EAGOC,WAAW,CAACC,MAAqB,EAAE;IACtC,KAAK,CAACA,MAAM,CAAC;IAAC;IAAA;IAAA;IAAA;IACd,IAAI,CAACC,MAAM,GAAGD,MAAM,CAACC,MAAM;IAC3B,IAAI,CAACC,MAAM,GAAGF,MAAM,CAACE,MAAM;EAC/B;AACJ;AAAC"}
1
+ {"version":3,"names":["Context","BaseContext","constructor","params","server","routes"],"sources":["Context.ts"],"sourcesContent":["import { Context as BaseContext, ContextParams as BaseContextParams } from \"@webiny/api\";\nimport { Context as BaseContextType } from \"~/types\";\n\nexport interface ContextParams extends BaseContextParams {\n server: BaseContextType[\"server\"];\n routes: BaseContextType[\"routes\"];\n}\n\nexport class Context extends BaseContext implements BaseContextType {\n public readonly server: BaseContextType[\"server\"];\n public readonly routes: BaseContextType[\"routes\"];\n // @ts-ignore\n public handlerClient: BaseContextType[\"handlerClient\"];\n // @ts-ignore\n public request: BaseContextType[\"request\"];\n // @ts-ignore\n public reply: BaseContextType[\"reply\"];\n\n public constructor(params: ContextParams) {\n super(params);\n this.server = params.server;\n this.routes = params.routes;\n }\n}\n"],"mappings":";;;;;;;;;;;AAAA;;AAQO,MAAMA,OAAN,SAAsBC,YAAtB,CAA6D;EAGhE;EAEA;EAEA;EAGOC,WAAW,CAACC,MAAD,EAAwB;IACtC,MAAMA,MAAN;IADsC;IAAA;IAAA;IAAA;IAAA;IAEtC,KAAKC,MAAL,GAAcD,MAAM,CAACC,MAArB;IACA,KAAKC,MAAL,GAAcF,MAAM,CAACE,MAArB;EACH;;AAd+D"}
package/fastify.js CHANGED
@@ -1,23 +1,44 @@
1
1
  "use strict";
2
2
 
3
3
  var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault").default;
4
+
4
5
  Object.defineProperty(exports, "__esModule", {
5
6
  value: true
6
7
  });
7
8
  exports.createHandler = void 0;
9
+
8
10
  var _objectSpread2 = _interopRequireDefault(require("@babel/runtime/helpers/objectSpread2"));
11
+
9
12
  var _fastify = _interopRequireDefault(require("fastify"));
13
+
10
14
  var _utils = require("@webiny/utils");
15
+
11
16
  var _Context = require("./Context");
17
+
12
18
  var _error = _interopRequireDefault(require("@webiny/error"));
19
+
13
20
  var _RoutePlugin = require("./plugins/RoutePlugin");
21
+
14
22
  var _handlerClient = require("@webiny/handler-client");
23
+
15
24
  var _cookie = _interopRequireDefault(require("@fastify/cookie"));
25
+
26
+ var _compress = _interopRequireDefault(require("@fastify/compress"));
27
+
16
28
  var _middleware = require("./middleware");
29
+
17
30
  var _api = require("@webiny/api");
31
+
18
32
  var _BeforeHandlerPlugin = require("./plugins/BeforeHandlerPlugin");
33
+
19
34
  var _HandlerResultPlugin = require("./plugins/HandlerResultPlugin");
35
+
20
36
  var _HandlerErrorPlugin = require("./plugins/HandlerErrorPlugin");
37
+
38
+ var _ModifyFastifyPlugin = require("./plugins/ModifyFastifyPlugin");
39
+
40
+ var _HandlerOnRequestPlugin = require("./plugins/HandlerOnRequestPlugin");
41
+
21
42
  const DEFAULT_HEADERS = (0, _objectSpread2.default)({
22
43
  "Cache-Control": "no-store",
23
44
  "Content-Type": "application/json; charset=utf-8",
@@ -25,31 +46,56 @@ const DEFAULT_HEADERS = (0, _objectSpread2.default)({
25
46
  "Access-Control-Allow-Headers": "*",
26
47
  "Access-Control-Allow-Methods": "OPTIONS,POST,GET,DELETE,PUT,PATCH"
27
48
  }, (0, _utils.getWebinyVersionHeaders)());
49
+
28
50
  const getDefaultHeaders = routes => {
29
51
  /**
30
52
  * If we are accepting all headers, just output that one.
31
53
  */
32
54
  const keys = Object.keys(routes);
33
55
  const all = keys.every(key => routes[key].length > 0);
56
+
34
57
  if (all) {
35
58
  return (0, _objectSpread2.default)((0, _objectSpread2.default)({}, DEFAULT_HEADERS), {}, {
36
59
  "Access-Control-Allow-Methods": "*"
37
60
  });
38
61
  }
62
+
39
63
  return (0, _objectSpread2.default)((0, _objectSpread2.default)({}, DEFAULT_HEADERS), {}, {
40
- "Access-Control-Allow-Methods": keys.filter(key => {
41
- const type = key;
64
+ "Access-Control-Allow-Methods": keys.filter(type => {
42
65
  if (!routes[type] || Array.isArray(routes[type]) === false) {
43
66
  return false;
44
67
  }
68
+
45
69
  return routes[type].length > 0;
46
70
  }).sort().join(",")
47
71
  });
48
72
  };
73
+
74
+ const stringifyError = error => {
75
+ var _error$constructor;
76
+
77
+ const {
78
+ name,
79
+ message,
80
+ code,
81
+ stack,
82
+ data
83
+ } = error;
84
+ return JSON.stringify((0, _objectSpread2.default)((0, _objectSpread2.default)({}, error), {}, {
85
+ constructorName: ((_error$constructor = error.constructor) === null || _error$constructor === void 0 ? void 0 : _error$constructor.name) || "UnknownError",
86
+ name: name || "No error name",
87
+ message: message || "No error message",
88
+ code: code || "NO_CODE",
89
+ data,
90
+ stack: process.env.DEBUG === "true" ? stack : "Turn on the debug flag to see the stack."
91
+ }));
92
+ };
93
+
49
94
  const OPTIONS_HEADERS = {
50
95
  "Access-Control-Max-Age": "86400",
51
96
  "Cache-Control": "public, max-age=86400"
52
97
  };
98
+
53
99
  const createHandler = params => {
54
100
  const definedRoutes = {
55
101
  POST: [],
@@ -69,15 +115,20 @@ const createHandler = params => {
69
115
  TRACE: [],
70
116
  UNLOCK: []
71
117
  };
118
+
72
119
  const throwOnDefinedRoute = (type, path, options) => {
73
120
  if (type === "ALL") {
74
- const all = Object.keys(definedRoutes).some(key => {
121
+ const all = Object.keys(definedRoutes).find(key => {
75
122
  const routes = definedRoutes[key];
76
123
  return routes.includes(path);
77
124
  });
125
+
78
126
  if (!all) {
79
127
  return;
80
128
  }
129
+
130
+ console.log(`Error while registering onAll route. One of the routes is already defined.`);
131
+ console.log(JSON.stringify(all));
81
132
  throw new _error.default(`You cannot override a route with onAll() method, please remove unnecessary route from the system.`, "OVERRIDE_ROUTE_ERROR", {
82
133
  type,
83
134
  path
@@ -87,46 +138,80 @@ const createHandler = params => {
87
138
  } else if ((options === null || options === void 0 ? void 0 : options.override) === true) {
88
139
  return;
89
140
  }
141
+
142
+ console.log(`Error while trying to override route: [${type}] ${path}`);
90
143
  throw new _error.default(`When you are trying to override existing route, you must send "override" parameter when adding that route.`, "OVERRIDE_ROUTE_ERROR", {
91
144
  type,
92
145
  path
93
146
  });
94
147
  };
95
- const addDefinedRoute = (inputType, path) => {
96
- const type = inputType.toUpperCase();
148
+
149
+ const addDefinedRoute = (type, path) => {
97
150
  if (!definedRoutes[type]) {
98
151
  return;
99
152
  } else if (definedRoutes[type].includes(path)) {
100
153
  return;
101
154
  }
155
+
102
156
  definedRoutes[type].push(path);
103
157
  };
104
158
  /**
105
159
  * We must attach the server to our internal context if we want to have it accessible.
106
160
  */
161
+
162
+
107
163
  const app = (0, _fastify.default)((0, _objectSpread2.default)({}, params.options || {}));
108
164
  /**
109
165
  * We need to register routes in our system so we can output headers later on and dissallow overriding routes.
110
166
  */
167
+
111
168
  app.addHook("onRoute", route => {
112
169
  const method = route.method;
170
+
113
171
  if (Array.isArray(method)) {
114
172
  for (const m of method) {
115
173
  addDefinedRoute(m, route.path);
116
174
  }
175
+
117
176
  return;
118
177
  }
178
+
119
179
  addDefinedRoute(method, route.path);
120
180
  });
121
181
  /**
182
+ * ############################
183
+ * Register the Fastify plugins.
184
+ */
185
+
186
+ /**
187
+ * Package @fastify/cookie
122
188
  *
189
+ * https://github.com/fastify/fastify-cookie
123
190
  */
191
+
124
192
  app.register(_cookie.default, {
125
193
  parseOptions: {} // options for parsing cookies
194
+
195
+ });
196
+ /**
197
+ * Package @fastify/compress
198
+ *
199
+ * https://github.com/fastify/fastify-compress
200
+ */
201
+
202
+ app.register(_compress.default, {
203
+ global: true,
204
+ threshold: 1024,
205
+ onUnsupportedEncoding: (encoding, _, reply) => {
206
+ reply.code(406);
207
+ return `We do not support the ${encoding} encoding.`;
208
+ },
209
+ inflateIfDeflated: true
126
210
  });
127
211
  /**
128
212
  * Route helpers - mostly for users.
129
213
  */
214
+
130
215
  const routes = {
131
216
  defined: definedRoutes,
132
217
  onPost: (path, handler, options) => {
@@ -162,25 +247,35 @@ const createHandler = params => {
162
247
  app.head(path, handler);
163
248
  }
164
249
  };
165
- const context = new _Context.Context({
166
- plugins: [
167
- /**
168
- * We must have handlerClient by default.
169
- * And it must be one of the first context plugins applied.
170
- */
171
- (0, _handlerClient.createHandlerClient)(), ...(params.plugins || [])],
172
- /**
173
- * Inserted via webpack on build time.
174
- */
175
- WEBINY_VERSION: process.env.WEBINY_VERSION,
176
- server: app,
177
- routes
178
- });
250
+ let context;
251
+
252
+ try {
253
+ context = new _Context.Context({
254
+ plugins: [
255
+ /**
256
+ * We must have handlerClient by default.
257
+ * And it must be one of the first context plugins applied.
258
+ */
259
+ (0, _handlerClient.createHandlerClient)(), ...(params.plugins || [])],
260
+
261
+ /**
262
+ * Inserted via webpack on build time.
263
+ */
264
+ WEBINY_VERSION: process.env.WEBINY_VERSION,
265
+ server: app,
266
+ routes
267
+ });
268
+ } catch (ex) {
269
+ console.log(`Error while constructing the Context.`);
270
+ console.log(stringifyError(ex));
271
+ throw ex;
272
+ }
179
273
  /**
180
- * We are attaching our custom context to webiny variable on the fastify app so it is accessible everywhere
274
+ * We are attaching our custom context to webiny variable on the fastify app, so it is accessible everywhere.
181
275
  */
182
- app.decorate("webiny", context);
183
276
 
277
+
278
+ app.decorate("webiny", context);
184
279
  /**
185
280
  * We have few types of triggers:
186
281
  * * Events - EventPlugin
@@ -188,80 +283,201 @@ const createHandler = params => {
188
283
  *
189
284
  * Routes are registered in fastify but events must be handled in package which implements cloud specific methods.
190
285
  */
191
- const routePlugins = app.webiny.plugins.byType(_RoutePlugin.RoutePlugin.type);
192
286
 
287
+ const routePlugins = app.webiny.plugins.byType(_RoutePlugin.RoutePlugin.type);
193
288
  /**
194
289
  * Add routes to the system.
195
290
  */
196
- for (const plugin of routePlugins) {
197
- plugin.cb((0, _objectSpread2.default)((0, _objectSpread2.default)({}, app.webiny.routes), {}, {
198
- context: app.webiny
199
- }));
200
- }
201
291
 
292
+ let routePluginName;
293
+
294
+ try {
295
+ for (const plugin of routePlugins) {
296
+ routePluginName = plugin.name;
297
+ plugin.cb((0, _objectSpread2.default)((0, _objectSpread2.default)({}, app.webiny.routes), {}, {
298
+ context: app.webiny
299
+ }));
300
+ }
301
+ } catch (ex) {
302
+ console.log(`Error while running the "RoutePlugin" ${routePluginName ? `(${routePluginName})` : ""} plugin in the beginning of the "createHandler" callable.`);
303
+ console.log(stringifyError(ex));
304
+ throw ex;
305
+ }
202
306
  /**
203
307
  * On every request we add default headers, which can be changed later.
204
308
  * Also, if it is an options request, we skip everything after this hook and output options headers.
205
309
  */
310
+
311
+
206
312
  app.addHook("onRequest", async (request, reply) => {
313
+ /**
314
+ * Our default headers are always set. Users can override them.
315
+ */
207
316
  const defaultHeaders = getDefaultHeaders(definedRoutes);
208
317
  reply.headers(defaultHeaders);
318
+ /**
319
+ * Users can define their own custom handlers for the onRequest event - so let's run them first.
320
+ */
321
+
322
+ const plugins = app.webiny.plugins.byType(_HandlerOnRequestPlugin.HandlerOnRequestPlugin.type);
323
+ let name;
324
+
325
+ try {
326
+ for (const plugin of plugins) {
327
+ name = plugin.name;
328
+ const result = await plugin.exec(request, reply);
329
+
330
+ if (result === false) {
331
+ return;
332
+ }
333
+ }
334
+ } catch (ex) {
335
+ console.log(`Error while running the "HandlerOnRequestPlugin" ${name ? `(${name})` : ""} plugin in the onRequest hook.`);
336
+ console.log(stringifyError(ex));
337
+ throw ex;
338
+ }
339
+ /**
340
+ * When we receive the OPTIONS request, we end it before it goes any further as there is no need for anything to run after this - at least for our use cases.
341
+ *
342
+ * Users can prevent this by creating their own HandlerOnRequestPlugin and returning false as the result of the callable.
343
+ */
344
+
345
+
209
346
  if (request.method !== "OPTIONS") {
210
347
  return;
211
348
  }
212
- const raw = reply.code(204).hijack().raw;
213
- const headers = (0, _objectSpread2.default)((0, _objectSpread2.default)({}, defaultHeaders), OPTIONS_HEADERS);
214
- for (const key in headers) {
215
- raw.setHeader(key, headers[key]);
349
+
350
+ if (reply.sent) {
351
+ /**
352
+ * At this point throwing an exception will not do anything with the response. So just log it.
353
+ */
354
+ console.log(JSON.stringify({
355
+ message: `Output was already sent. Please check custom plugins of type "HandlerOnRequestPlugin".`,
356
+ explanation: "This error can happen if the user plugin ended the reply, but did not return false as response."
357
+ }));
358
+ return;
216
359
  }
217
- raw.end("");
360
+
361
+ reply.headers((0, _objectSpread2.default)((0, _objectSpread2.default)({}, defaultHeaders), OPTIONS_HEADERS)).code(204).send("").hijack();
218
362
  });
219
- app.addHook("preParsing", async request => {
363
+ app.addHook("preParsing", async (request, reply) => {
220
364
  app.webiny.request = request;
365
+ app.webiny.reply = reply;
221
366
  const plugins = app.webiny.plugins.byType(_api.ContextPlugin.type);
222
- for (const plugin of plugins) {
223
- await plugin.apply(app.webiny);
367
+ let name;
368
+
369
+ try {
370
+ for (const plugin of plugins) {
371
+ name = plugin.name;
372
+ await plugin.apply(app.webiny);
373
+ }
374
+ } catch (ex) {
375
+ console.log(`Error while running the "ContextPlugin" ${name ? `(${name})` : ""} plugin in the preParsing hook.`);
376
+ console.log(stringifyError(ex));
377
+ throw ex;
224
378
  }
225
379
  });
226
380
  /**
227
381
  *
228
382
  */
383
+
229
384
  app.addHook("preHandler", async () => {
230
385
  const plugins = app.webiny.plugins.byType(_BeforeHandlerPlugin.BeforeHandlerPlugin.type);
231
- for (const plugin of plugins) {
232
- await plugin.apply(app.webiny);
386
+ let name;
387
+
388
+ try {
389
+ for (const plugin of plugins) {
390
+ name = plugin.name;
391
+ await plugin.apply(app.webiny);
392
+ }
393
+ } catch (ex) {
394
+ console.log(`Error while running the "BeforeHandlerPlugin" ${name ? `(${name})` : ""} plugin in the preHandler hook.`);
395
+ console.log(stringifyError(ex));
396
+ throw ex;
233
397
  }
234
398
  });
235
-
236
399
  /**
237
400
  *
238
401
  */
402
+
239
403
  const preSerialization = async (_, __, payload) => {
240
404
  const plugins = app.webiny.plugins.byType(_HandlerResultPlugin.HandlerResultPlugin.type);
241
- for (const plugin of plugins) {
242
- await plugin.handle(app.webiny, payload);
405
+ let name;
406
+
407
+ try {
408
+ for (const plugin of plugins) {
409
+ name = plugin.name;
410
+ await plugin.handle(app.webiny, payload);
411
+ }
412
+ } catch (ex) {
413
+ console.log(`Error while running the "HandlerResultPlugin" ${name ? `(${name})` : ""} plugin in the preSerialization hook.`);
414
+ console.log(stringifyError(ex));
415
+ throw ex;
243
416
  }
417
+
244
418
  return payload;
245
419
  };
420
+
246
421
  app.addHook("preSerialization", preSerialization);
422
+ app.setErrorHandler(async (error, request, reply) => {
423
+ return reply.status(500).headers({
424
+ "Cache-Control": "no-store"
425
+ }).send(
426
+ /**
427
+ * When we are sending the error in the response, we cannot send the whole error object, as it might contain some sensitive data.
428
+ */
429
+ JSON.stringify({
430
+ message: error.message,
431
+ code: error.code,
432
+ data: error.data
433
+ }));
434
+ });
247
435
  app.addHook("onError", async (_, reply, error) => {
248
436
  const plugins = app.webiny.plugins.byType(_HandlerErrorPlugin.HandlerErrorPlugin.type);
249
- // Log error to cloud, as these can be extremely annoying to debug!
437
+ /**
438
+ * Log error to cloud, as these can be extremely annoying to debug!
439
+ */
440
+
250
441
  console.log("@webiny/handler");
251
- console.log(JSON.stringify((0, _objectSpread2.default)((0, _objectSpread2.default)({}, error || {}), {}, {
252
- message: error === null || error === void 0 ? void 0 : error.message,
253
- code: error === null || error === void 0 ? void 0 : error.code
254
- })));
442
+ console.log(stringifyError(error));
443
+ reply.status(500).headers({
444
+ "Cache-Control": "no-store"
445
+ }).send(
446
+ /**
447
+ * When we are sending the error in the response, we cannot send the whole error object, as it might contain some sensitive data.
448
+ */
449
+ JSON.stringify({
450
+ message: error.message,
451
+ code: error.code,
452
+ data: error.data
453
+ }));
255
454
  const handler = (0, _middleware.middleware)(plugins.map(pl => {
256
455
  return (context, error, next) => {
257
456
  return pl.handle(context, error, next);
258
457
  };
259
458
  }));
260
459
  await handler(app.webiny, error);
261
- return reply.headers({
262
- "Cache-Control": "no-store"
263
- }).status(500);
460
+ return reply;
264
461
  });
462
+ /**
463
+ * With these plugins we give users possibility to do anything they want on our fastify instance.
464
+ */
465
+
466
+ const modifyPlugins = app.webiny.plugins.byType(_ModifyFastifyPlugin.ModifyFastifyPlugin.type);
467
+ let modifyFastifyPluginName;
468
+
469
+ try {
470
+ for (const plugin of modifyPlugins) {
471
+ modifyFastifyPluginName = plugin.name;
472
+ plugin.modify(app);
473
+ }
474
+ } catch (ex) {
475
+ console.log(`Error while running the "ModifyFastifyPlugin" ${modifyFastifyPluginName ? `(${modifyFastifyPluginName})` : ""} plugin in the end of the "createHandler" callable.`);
476
+ console.log(stringifyError(ex));
477
+ throw ex;
478
+ }
479
+
265
480
  return app;
266
481
  };
482
+
267
483
  exports.createHandler = createHandler;
package/fastify.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"names":["DEFAULT_HEADERS","getWebinyVersionHeaders","getDefaultHeaders","routes","keys","Object","all","every","key","length","filter","type","Array","isArray","sort","join","OPTIONS_HEADERS","createHandler","params","definedRoutes","POST","GET","OPTIONS","DELETE","PATCH","PUT","HEAD","COPY","LOCK","MKCOL","MOVE","PROPFIND","PROPPATCH","SEARCH","TRACE","UNLOCK","throwOnDefinedRoute","path","options","some","includes","WebinyError","override","addDefinedRoute","inputType","toUpperCase","push","app","fastify","addHook","route","method","m","register","fastifyCookies","parseOptions","defined","onPost","handler","post","onGet","get","onOptions","onDelete","delete","onPatch","patch","onPut","put","onAll","onHead","head","context","Context","plugins","createHandlerClient","WEBINY_VERSION","process","env","server","decorate","routePlugins","webiny","byType","RoutePlugin","plugin","cb","request","reply","defaultHeaders","headers","raw","code","hijack","setHeader","end","ContextPlugin","apply","BeforeHandlerPlugin","preSerialization","_","__","payload","HandlerResultPlugin","handle","error","HandlerErrorPlugin","console","log","JSON","stringify","message","middleware","map","pl","next","status"],"sources":["fastify.ts"],"sourcesContent":["import { PluginCollection } from \"@webiny/plugins/types\";\nimport fastify, {\n FastifyServerOptions as ServerOptions,\n preSerializationAsyncHookHandler\n} from \"fastify\";\nimport { getWebinyVersionHeaders } from \"@webiny/utils\";\nimport { ContextRoutes, DefinedContextRoutes, RouteMethodOptions, HTTPMethods } from \"~/types\";\nimport { Context } from \"~/Context\";\nimport WebinyError from \"@webiny/error\";\nimport { RoutePlugin } from \"./plugins/RoutePlugin\";\nimport { createHandlerClient } from \"@webiny/handler-client\";\nimport fastifyCookies from \"@fastify/cookie\";\nimport { middleware } from \"~/middleware\";\nimport { ContextPlugin } from \"@webiny/api\";\nimport { BeforeHandlerPlugin } from \"./plugins/BeforeHandlerPlugin\";\nimport { HandlerResultPlugin } from \"./plugins/HandlerResultPlugin\";\nimport { HandlerErrorPlugin } from \"./plugins/HandlerErrorPlugin\";\n\nconst DEFAULT_HEADERS: Record<string, string> = {\n \"Cache-Control\": \"no-store\",\n \"Content-Type\": \"application/json; charset=utf-8\",\n \"Access-Control-Allow-Origin\": \"*\",\n \"Access-Control-Allow-Headers\": \"*\",\n \"Access-Control-Allow-Methods\": \"OPTIONS,POST,GET,DELETE,PUT,PATCH\",\n ...getWebinyVersionHeaders()\n};\n\nconst getDefaultHeaders = (routes: DefinedContextRoutes): Record<string, string> => {\n /**\n * If we are accepting all headers, just output that one.\n */\n const keys = Object.keys(routes);\n const all = keys.every(key => routes[key as HTTPMethods].length > 0);\n if (all) {\n return {\n ...DEFAULT_HEADERS,\n \"Access-Control-Allow-Methods\": \"*\"\n };\n }\n return {\n ...DEFAULT_HEADERS,\n \"Access-Control-Allow-Methods\": keys\n .filter(key => {\n const type = key as unknown as HTTPMethods;\n if (!routes[type] || Array.isArray(routes[type]) === false) {\n return false;\n }\n return routes[type].length > 0;\n })\n .sort()\n .join(\",\")\n };\n};\n\nconst OPTIONS_HEADERS: Record<string, string> = {\n \"Access-Control-Max-Age\": \"86400\",\n \"Cache-Control\": \"public, max-age=86400\"\n};\n\nexport interface CreateHandlerParams {\n plugins: PluginCollection;\n options?: ServerOptions;\n}\n\nexport const createHandler = (params: CreateHandlerParams) => {\n const definedRoutes: DefinedContextRoutes = {\n POST: [],\n GET: [],\n OPTIONS: [],\n DELETE: [],\n PATCH: [],\n PUT: [],\n HEAD: [],\n COPY: [],\n LOCK: [],\n MKCOL: [],\n MOVE: [],\n PROPFIND: [],\n PROPPATCH: [],\n SEARCH: [],\n TRACE: [],\n UNLOCK: []\n };\n\n const throwOnDefinedRoute = (\n type: HTTPMethods | \"ALL\",\n path: string,\n options?: RouteMethodOptions\n ): void => {\n if (type === \"ALL\") {\n const all = Object.keys(definedRoutes).some(key => {\n const routes = definedRoutes[key as HTTPMethods];\n return routes.includes(path);\n });\n if (!all) {\n return;\n }\n throw new WebinyError(\n `You cannot override a route with onAll() method, please remove unnecessary route from the system.`,\n \"OVERRIDE_ROUTE_ERROR\",\n {\n type,\n path\n }\n );\n } else if (definedRoutes[type].includes(path) === false) {\n return;\n } else if (options?.override === true) {\n return;\n }\n throw new WebinyError(\n `When you are trying to override existing route, you must send \"override\" parameter when adding that route.`,\n \"OVERRIDE_ROUTE_ERROR\",\n {\n type,\n path\n }\n );\n };\n\n const addDefinedRoute = (inputType: HTTPMethods, path: string): void => {\n const type = (inputType as string).toUpperCase() as HTTPMethods;\n if (!definedRoutes[type]) {\n return;\n } else if (definedRoutes[type].includes(path)) {\n return;\n }\n definedRoutes[type].push(path);\n };\n /**\n * We must attach the server to our internal context if we want to have it accessible.\n */\n const app = fastify({\n ...(params.options || {})\n });\n /**\n * We need to register routes in our system so we can output headers later on and dissallow overriding routes.\n */\n app.addHook(\"onRoute\", route => {\n const method = route.method;\n if (Array.isArray(method)) {\n for (const m of method) {\n addDefinedRoute(m, route.path);\n }\n return;\n }\n addDefinedRoute(method, route.path);\n });\n /**\n *\n */\n app.register(fastifyCookies, {\n parseOptions: {} // options for parsing cookies\n });\n /**\n * Route helpers - mostly for users.\n */\n const routes: ContextRoutes = {\n defined: definedRoutes,\n onPost: (path, handler, options) => {\n throwOnDefinedRoute(\"POST\", path, options);\n app.post(path, handler);\n },\n onGet: (path, handler, options) => {\n throwOnDefinedRoute(\"GET\", path, options);\n app.get(path, handler);\n },\n onOptions: (path, handler, options) => {\n throwOnDefinedRoute(\"OPTIONS\", path, options);\n app.options(path, handler);\n },\n onDelete: (path, handler, options) => {\n throwOnDefinedRoute(\"DELETE\", path, options);\n app.delete(path, handler);\n },\n onPatch: (path, handler, options) => {\n throwOnDefinedRoute(\"PATCH\", path, options);\n app.patch(path, handler);\n },\n onPut: (path, handler, options) => {\n throwOnDefinedRoute(\"PUT\", path, options);\n app.put(path, handler);\n },\n onAll: (path, handler, options) => {\n throwOnDefinedRoute(\"ALL\", path, options);\n app.all(path, handler);\n },\n onHead: (path, handler, options) => {\n throwOnDefinedRoute(\"HEAD\", path, options);\n app.head(path, handler);\n }\n };\n const context = new Context({\n plugins: [\n /**\n * We must have handlerClient by default.\n * And it must be one of the first context plugins applied.\n */\n createHandlerClient(),\n ...(params.plugins || [])\n ],\n /**\n * Inserted via webpack on build time.\n */\n WEBINY_VERSION: process.env.WEBINY_VERSION as string,\n server: app,\n routes\n });\n /**\n * We are attaching our custom context to webiny variable on the fastify app so it is accessible everywhere\n */\n app.decorate(\"webiny\", context);\n\n /**\n * We have few types of triggers:\n * * Events - EventPlugin\n * * Routes - RoutePlugin\n *\n * Routes are registered in fastify but events must be handled in package which implements cloud specific methods.\n */\n const routePlugins = app.webiny.plugins.byType<RoutePlugin>(RoutePlugin.type);\n\n /**\n * Add routes to the system.\n */\n for (const plugin of routePlugins) {\n plugin.cb({\n ...app.webiny.routes,\n context: app.webiny\n });\n }\n\n /**\n * On every request we add default headers, which can be changed later.\n * Also, if it is an options request, we skip everything after this hook and output options headers.\n */\n app.addHook(\"onRequest\", async (request, reply) => {\n const defaultHeaders = getDefaultHeaders(definedRoutes);\n reply.headers(defaultHeaders);\n if (request.method !== \"OPTIONS\") {\n return;\n }\n const raw = reply.code(204).hijack().raw;\n const headers = { ...defaultHeaders, ...OPTIONS_HEADERS };\n for (const key in headers) {\n raw.setHeader(key, headers[key]);\n }\n\n raw.end(\"\");\n });\n\n app.addHook(\"preParsing\", async request => {\n app.webiny.request = request;\n const plugins = app.webiny.plugins.byType<ContextPlugin>(ContextPlugin.type);\n for (const plugin of plugins) {\n await plugin.apply(app.webiny);\n }\n });\n /**\n *\n */\n app.addHook(\"preHandler\", async () => {\n const plugins = app.webiny.plugins.byType<BeforeHandlerPlugin>(BeforeHandlerPlugin.type);\n for (const plugin of plugins) {\n await plugin.apply(app.webiny);\n }\n });\n\n /**\n *\n */\n const preSerialization: preSerializationAsyncHookHandler<any> = async (_, __, payload) => {\n const plugins = app.webiny.plugins.byType<HandlerResultPlugin>(HandlerResultPlugin.type);\n for (const plugin of plugins) {\n await plugin.handle(app.webiny, payload);\n }\n return payload;\n };\n\n app.addHook(\"preSerialization\", preSerialization);\n\n app.addHook(\"onError\", async (_, reply, error) => {\n const plugins = app.webiny.plugins.byType<HandlerErrorPlugin>(HandlerErrorPlugin.type);\n // Log error to cloud, as these can be extremely annoying to debug!\n console.log(\"@webiny/handler\");\n console.log(\n JSON.stringify({\n ...(error || {}),\n message: error?.message,\n code: error?.code\n })\n );\n const handler = middleware(\n plugins.map(pl => {\n return (context: Context, error: Error, next: Function) => {\n return pl.handle(context, error, next);\n };\n })\n );\n await handler(app.webiny, error);\n\n return reply\n .headers({\n \"Cache-Control\": \"no-store\"\n })\n .status(500);\n });\n\n return app;\n};\n"],"mappings":"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"}
1
+ {"version":3,"names":["DEFAULT_HEADERS","getWebinyVersionHeaders","getDefaultHeaders","routes","keys","Object","all","every","key","length","filter","type","Array","isArray","sort","join","stringifyError","error","name","message","code","stack","data","JSON","stringify","constructorName","constructor","process","env","DEBUG","OPTIONS_HEADERS","createHandler","params","definedRoutes","POST","GET","OPTIONS","DELETE","PATCH","PUT","HEAD","COPY","LOCK","MKCOL","MOVE","PROPFIND","PROPPATCH","SEARCH","TRACE","UNLOCK","throwOnDefinedRoute","path","options","find","includes","console","log","WebinyError","override","addDefinedRoute","push","app","fastify","addHook","route","method","m","register","fastifyCookie","parseOptions","fastifyCompress","global","threshold","onUnsupportedEncoding","encoding","_","reply","inflateIfDeflated","defined","onPost","handler","post","onGet","get","onOptions","onDelete","delete","onPatch","patch","onPut","put","onAll","onHead","head","context","Context","plugins","createHandlerClient","WEBINY_VERSION","server","ex","decorate","routePlugins","webiny","byType","RoutePlugin","routePluginName","plugin","cb","request","defaultHeaders","headers","HandlerOnRequestPlugin","result","exec","sent","explanation","send","hijack","ContextPlugin","apply","BeforeHandlerPlugin","preSerialization","__","payload","HandlerResultPlugin","handle","setErrorHandler","status","HandlerErrorPlugin","middleware","map","pl","next","modifyPlugins","ModifyFastifyPlugin","modifyFastifyPluginName","modify"],"sources":["fastify.ts"],"sourcesContent":["import { PluginCollection } from \"@webiny/plugins/types\";\nimport fastify, {\n FastifyServerOptions as ServerOptions,\n preSerializationAsyncHookHandler\n} from \"fastify\";\nimport { getWebinyVersionHeaders } from \"@webiny/utils\";\nimport { ContextRoutes, DefinedContextRoutes, RouteMethodOptions, HTTPMethods } from \"~/types\";\nimport { Context } from \"~/Context\";\nimport WebinyError from \"@webiny/error\";\nimport { RoutePlugin } from \"./plugins/RoutePlugin\";\nimport { createHandlerClient } from \"@webiny/handler-client\";\nimport fastifyCookie from \"@fastify/cookie\";\nimport fastifyCompress from \"@fastify/compress\";\nimport { middleware } from \"~/middleware\";\nimport { ContextPlugin } from \"@webiny/api\";\nimport { BeforeHandlerPlugin } from \"./plugins/BeforeHandlerPlugin\";\nimport { HandlerResultPlugin } from \"./plugins/HandlerResultPlugin\";\nimport { HandlerErrorPlugin } from \"./plugins/HandlerErrorPlugin\";\nimport { ModifyFastifyPlugin } from \"~/plugins/ModifyFastifyPlugin\";\nimport { HandlerOnRequestPlugin } from \"~/plugins/HandlerOnRequestPlugin\";\n\nconst DEFAULT_HEADERS: Record<string, string> = {\n \"Cache-Control\": \"no-store\",\n \"Content-Type\": \"application/json; charset=utf-8\",\n \"Access-Control-Allow-Origin\": \"*\",\n \"Access-Control-Allow-Headers\": \"*\",\n \"Access-Control-Allow-Methods\": \"OPTIONS,POST,GET,DELETE,PUT,PATCH\",\n ...getWebinyVersionHeaders()\n};\n\nconst getDefaultHeaders = (routes: DefinedContextRoutes): Record<string, string> => {\n /**\n * If we are accepting all headers, just output that one.\n */\n const keys = Object.keys(routes) as HTTPMethods[];\n const all = keys.every(key => routes[key].length > 0);\n if (all) {\n return {\n ...DEFAULT_HEADERS,\n \"Access-Control-Allow-Methods\": \"*\"\n };\n }\n return {\n ...DEFAULT_HEADERS,\n \"Access-Control-Allow-Methods\": keys\n .filter(type => {\n if (!routes[type] || Array.isArray(routes[type]) === false) {\n return false;\n }\n return routes[type].length > 0;\n })\n .sort()\n .join(\",\")\n };\n};\n\nconst stringifyError = (error: Error) => {\n const { name, message, code, stack, data } = error as any;\n return JSON.stringify({\n ...error,\n constructorName: error.constructor?.name || \"UnknownError\",\n name: name || \"No error name\",\n message: message || \"No error message\",\n code: code || \"NO_CODE\",\n data,\n stack: process.env.DEBUG === \"true\" ? stack : \"Turn on the debug flag to see the stack.\"\n });\n};\n\nconst OPTIONS_HEADERS: Record<string, string> = {\n \"Access-Control-Max-Age\": \"86400\",\n \"Cache-Control\": \"public, max-age=86400\"\n};\n\nexport interface CreateHandlerParams {\n plugins: PluginCollection;\n options?: ServerOptions;\n}\n\nexport const createHandler = (params: CreateHandlerParams) => {\n const definedRoutes: DefinedContextRoutes = {\n POST: [],\n GET: [],\n OPTIONS: [],\n DELETE: [],\n PATCH: [],\n PUT: [],\n HEAD: [],\n COPY: [],\n LOCK: [],\n MKCOL: [],\n MOVE: [],\n PROPFIND: [],\n PROPPATCH: [],\n SEARCH: [],\n TRACE: [],\n UNLOCK: []\n };\n\n const throwOnDefinedRoute = (\n type: HTTPMethods | \"ALL\",\n path: string,\n options?: RouteMethodOptions\n ): void => {\n if (type === \"ALL\") {\n const all = Object.keys(definedRoutes).find(key => {\n const routes = definedRoutes[key as HTTPMethods];\n return routes.includes(path);\n });\n if (!all) {\n return;\n }\n console.log(\n `Error while registering onAll route. One of the routes is already defined.`\n );\n console.log(JSON.stringify(all));\n throw new WebinyError(\n `You cannot override a route with onAll() method, please remove unnecessary route from the system.`,\n \"OVERRIDE_ROUTE_ERROR\",\n {\n type,\n path\n }\n );\n } else if (definedRoutes[type].includes(path) === false) {\n return;\n } else if (options?.override === true) {\n return;\n }\n console.log(`Error while trying to override route: [${type}] ${path}`);\n throw new WebinyError(\n `When you are trying to override existing route, you must send \"override\" parameter when adding that route.`,\n \"OVERRIDE_ROUTE_ERROR\",\n {\n type,\n path\n }\n );\n };\n\n const addDefinedRoute = (type: HTTPMethods, path: string): void => {\n if (!definedRoutes[type]) {\n return;\n } else if (definedRoutes[type].includes(path)) {\n return;\n }\n definedRoutes[type].push(path);\n };\n /**\n * We must attach the server to our internal context if we want to have it accessible.\n */\n const app = fastify({\n ...(params.options || {})\n });\n /**\n * We need to register routes in our system so we can output headers later on and dissallow overriding routes.\n */\n app.addHook(\"onRoute\", route => {\n const method = route.method;\n if (Array.isArray(method)) {\n for (const m of method) {\n addDefinedRoute(m, route.path);\n }\n return;\n }\n addDefinedRoute(method, route.path);\n });\n /**\n * ############################\n * Register the Fastify plugins.\n */\n /**\n * Package @fastify/cookie\n *\n * https://github.com/fastify/fastify-cookie\n */\n app.register(fastifyCookie, {\n parseOptions: {} // options for parsing cookies\n });\n /**\n * Package @fastify/compress\n *\n * https://github.com/fastify/fastify-compress\n */\n app.register(fastifyCompress, {\n global: true,\n threshold: 1024,\n onUnsupportedEncoding: (encoding, _, reply) => {\n reply.code(406);\n return `We do not support the ${encoding} encoding.`;\n },\n inflateIfDeflated: true\n });\n /**\n * Route helpers - mostly for users.\n */\n const routes: ContextRoutes = {\n defined: definedRoutes,\n onPost: (path, handler, options) => {\n throwOnDefinedRoute(\"POST\", path, options);\n app.post(path, handler);\n },\n onGet: (path, handler, options) => {\n throwOnDefinedRoute(\"GET\", path, options);\n app.get(path, handler);\n },\n onOptions: (path, handler, options) => {\n throwOnDefinedRoute(\"OPTIONS\", path, options);\n app.options(path, handler);\n },\n onDelete: (path, handler, options) => {\n throwOnDefinedRoute(\"DELETE\", path, options);\n app.delete(path, handler);\n },\n onPatch: (path, handler, options) => {\n throwOnDefinedRoute(\"PATCH\", path, options);\n app.patch(path, handler);\n },\n onPut: (path, handler, options) => {\n throwOnDefinedRoute(\"PUT\", path, options);\n app.put(path, handler);\n },\n onAll: (path, handler, options) => {\n throwOnDefinedRoute(\"ALL\", path, options);\n app.all(path, handler);\n },\n onHead: (path, handler, options) => {\n throwOnDefinedRoute(\"HEAD\", path, options);\n app.head(path, handler);\n }\n };\n let context: Context;\n try {\n context = new Context({\n plugins: [\n /**\n * We must have handlerClient by default.\n * And it must be one of the first context plugins applied.\n */\n createHandlerClient(),\n ...(params.plugins || [])\n ],\n /**\n * Inserted via webpack on build time.\n */\n WEBINY_VERSION: process.env.WEBINY_VERSION as string,\n server: app,\n routes\n });\n } catch (ex) {\n console.log(`Error while constructing the Context.`);\n console.log(stringifyError(ex));\n throw ex;\n }\n\n /**\n * We are attaching our custom context to webiny variable on the fastify app, so it is accessible everywhere.\n */\n app.decorate(\"webiny\", context);\n\n /**\n * We have few types of triggers:\n * * Events - EventPlugin\n * * Routes - RoutePlugin\n *\n * Routes are registered in fastify but events must be handled in package which implements cloud specific methods.\n */\n const routePlugins = app.webiny.plugins.byType<RoutePlugin>(RoutePlugin.type);\n\n /**\n * Add routes to the system.\n */\n let routePluginName: string | undefined;\n try {\n for (const plugin of routePlugins) {\n routePluginName = plugin.name;\n plugin.cb({\n ...app.webiny.routes,\n context: app.webiny\n });\n }\n } catch (ex) {\n console.log(\n `Error while running the \"RoutePlugin\" ${\n routePluginName ? `(${routePluginName})` : \"\"\n } plugin in the beginning of the \"createHandler\" callable.`\n );\n console.log(stringifyError(ex));\n throw ex;\n }\n\n /**\n * On every request we add default headers, which can be changed later.\n * Also, if it is an options request, we skip everything after this hook and output options headers.\n */\n app.addHook(\"onRequest\", async (request, reply) => {\n /**\n * Our default headers are always set. Users can override them.\n */\n const defaultHeaders = getDefaultHeaders(definedRoutes);\n reply.headers(defaultHeaders);\n /**\n * Users can define their own custom handlers for the onRequest event - so let's run them first.\n */\n const plugins = app.webiny.plugins.byType<HandlerOnRequestPlugin>(\n HandlerOnRequestPlugin.type\n );\n\n let name: string | undefined;\n try {\n for (const plugin of plugins) {\n name = plugin.name;\n const result = await plugin.exec(request, reply);\n if (result === false) {\n return;\n }\n }\n } catch (ex) {\n console.log(\n `Error while running the \"HandlerOnRequestPlugin\" ${\n name ? `(${name})` : \"\"\n } plugin in the onRequest hook.`\n );\n console.log(stringifyError(ex));\n throw ex;\n }\n /**\n * When we receive the OPTIONS request, we end it before it goes any further as there is no need for anything to run after this - at least for our use cases.\n *\n * Users can prevent this by creating their own HandlerOnRequestPlugin and returning false as the result of the callable.\n */\n if (request.method !== \"OPTIONS\") {\n return;\n }\n\n if (reply.sent) {\n /**\n * At this point throwing an exception will not do anything with the response. So just log it.\n */\n console.log(\n JSON.stringify({\n message: `Output was already sent. Please check custom plugins of type \"HandlerOnRequestPlugin\".`,\n explanation:\n \"This error can happen if the user plugin ended the reply, but did not return false as response.\"\n })\n );\n return;\n }\n\n reply\n .headers({ ...defaultHeaders, ...OPTIONS_HEADERS })\n .code(204)\n .send(\"\")\n .hijack();\n });\n\n app.addHook(\"preParsing\", async (request, reply) => {\n app.webiny.request = request;\n app.webiny.reply = reply;\n const plugins = app.webiny.plugins.byType<ContextPlugin>(ContextPlugin.type);\n let name: string | undefined;\n try {\n for (const plugin of plugins) {\n name = plugin.name;\n await plugin.apply(app.webiny);\n }\n } catch (ex) {\n console.log(\n `Error while running the \"ContextPlugin\" ${\n name ? `(${name})` : \"\"\n } plugin in the preParsing hook.`\n );\n console.log(stringifyError(ex));\n throw ex;\n }\n });\n /**\n *\n */\n app.addHook(\"preHandler\", async () => {\n const plugins = app.webiny.plugins.byType<BeforeHandlerPlugin>(BeforeHandlerPlugin.type);\n let name: string | undefined;\n try {\n for (const plugin of plugins) {\n name = plugin.name;\n await plugin.apply(app.webiny);\n }\n } catch (ex) {\n console.log(\n `Error while running the \"BeforeHandlerPlugin\" ${\n name ? `(${name})` : \"\"\n } plugin in the preHandler hook.`\n );\n console.log(stringifyError(ex));\n throw ex;\n }\n });\n\n /**\n *\n */\n const preSerialization: preSerializationAsyncHookHandler<any> = async (_, __, payload) => {\n const plugins = app.webiny.plugins.byType<HandlerResultPlugin>(HandlerResultPlugin.type);\n let name: string | undefined;\n try {\n for (const plugin of plugins) {\n name = plugin.name;\n await plugin.handle(app.webiny, payload);\n }\n } catch (ex) {\n console.log(\n `Error while running the \"HandlerResultPlugin\" ${\n name ? `(${name})` : \"\"\n } plugin in the preSerialization hook.`\n );\n console.log(stringifyError(ex));\n throw ex;\n }\n return payload;\n };\n\n app.addHook(\"preSerialization\", preSerialization);\n\n app.setErrorHandler<WebinyError>(async (error, request, reply) => {\n return reply\n .status(500)\n .headers({\n \"Cache-Control\": \"no-store\"\n })\n .send(\n /**\n * When we are sending the error in the response, we cannot send the whole error object, as it might contain some sensitive data.\n */\n JSON.stringify({\n message: error.message,\n code: error.code,\n data: error.data\n })\n );\n });\n\n app.addHook(\"onError\", async (_, reply, error: any) => {\n const plugins = app.webiny.plugins.byType<HandlerErrorPlugin>(HandlerErrorPlugin.type);\n /**\n * Log error to cloud, as these can be extremely annoying to debug!\n */\n console.log(\"@webiny/handler\");\n console.log(stringifyError(error));\n\n reply\n .status(500)\n .headers({\n \"Cache-Control\": \"no-store\"\n })\n .send(\n /**\n * When we are sending the error in the response, we cannot send the whole error object, as it might contain some sensitive data.\n */\n JSON.stringify({\n message: error.message,\n code: error.code,\n data: error.data\n })\n );\n\n const handler = middleware(\n plugins.map(pl => {\n return (context: Context, error: Error, next: Function) => {\n return pl.handle(context, error, next);\n };\n })\n );\n await handler(app.webiny, error);\n\n return reply;\n });\n\n /**\n * With these plugins we give users possibility to do anything they want on our fastify instance.\n */\n const modifyPlugins = app.webiny.plugins.byType<ModifyFastifyPlugin>(ModifyFastifyPlugin.type);\n\n let modifyFastifyPluginName: string | undefined;\n try {\n for (const plugin of modifyPlugins) {\n modifyFastifyPluginName = plugin.name;\n plugin.modify(app);\n }\n } catch (ex) {\n console.log(\n `Error while running the \"ModifyFastifyPlugin\" ${\n modifyFastifyPluginName ? `(${modifyFastifyPluginName})` : \"\"\n } plugin in the end of the \"createHandler\" callable.`\n );\n console.log(stringifyError(ex));\n throw ex;\n }\n\n return app;\n};\n"],"mappings":";;;;;;;;;;;AACA;;AAIA;;AAEA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AAEA,MAAMA,eAAuC;EACzC,iBAAiB,UADwB;EAEzC,gBAAgB,iCAFyB;EAGzC,+BAA+B,GAHU;EAIzC,gCAAgC,GAJS;EAKzC,gCAAgC;AALS,GAMtC,IAAAC,8BAAA,GANsC,CAA7C;;AASA,MAAMC,iBAAiB,GAAIC,MAAD,IAA0D;EAChF;AACJ;AACA;EACI,MAAMC,IAAI,GAAGC,MAAM,CAACD,IAAP,CAAYD,MAAZ,CAAb;EACA,MAAMG,GAAG,GAAGF,IAAI,CAACG,KAAL,CAAWC,GAAG,IAAIL,MAAM,CAACK,GAAD,CAAN,CAAYC,MAAZ,GAAqB,CAAvC,CAAZ;;EACA,IAAIH,GAAJ,EAAS;IACL,mEACON,eADP;MAEI,gCAAgC;IAFpC;EAIH;;EACD,mEACOA,eADP;IAEI,gCAAgCI,IAAI,CAC/BM,MAD2B,CACpBC,IAAI,IAAI;MACZ,IAAI,CAACR,MAAM,CAACQ,IAAD,CAAP,IAAiBC,KAAK,CAACC,OAAN,CAAcV,MAAM,CAACQ,IAAD,CAApB,MAAgC,KAArD,EAA4D;QACxD,OAAO,KAAP;MACH;;MACD,OAAOR,MAAM,CAACQ,IAAD,CAAN,CAAaF,MAAb,GAAsB,CAA7B;IACH,CAN2B,EAO3BK,IAP2B,GAQ3BC,IAR2B,CAQtB,GARsB;EAFpC;AAYH,CAxBD;;AA0BA,MAAMC,cAAc,GAAIC,KAAD,IAAkB;EAAA;;EACrC,MAAM;IAAEC,IAAF;IAAQC,OAAR;IAAiBC,IAAjB;IAAuBC,KAAvB;IAA8BC;EAA9B,IAAuCL,KAA7C;EACA,OAAOM,IAAI,CAACC,SAAL,6DACAP,KADA;IAEHQ,eAAe,EAAE,uBAAAR,KAAK,CAACS,WAAN,0EAAmBR,IAAnB,KAA2B,cAFzC;IAGHA,IAAI,EAAEA,IAAI,IAAI,eAHX;IAIHC,OAAO,EAAEA,OAAO,IAAI,kBAJjB;IAKHC,IAAI,EAAEA,IAAI,IAAI,SALX;IAMHE,IANG;IAOHD,KAAK,EAAEM,OAAO,CAACC,GAAR,CAAYC,KAAZ,KAAsB,MAAtB,GAA+BR,KAA/B,GAAuC;EAP3C,GAAP;AASH,CAXD;;AAaA,MAAMS,eAAuC,GAAG;EAC5C,0BAA0B,OADkB;EAE5C,iBAAiB;AAF2B,CAAhD;;AAUO,MAAMC,aAAa,GAAIC,MAAD,IAAiC;EAC1D,MAAMC,aAAmC,GAAG;IACxCC,IAAI,EAAE,EADkC;IAExCC,GAAG,EAAE,EAFmC;IAGxCC,OAAO,EAAE,EAH+B;IAIxCC,MAAM,EAAE,EAJgC;IAKxCC,KAAK,EAAE,EALiC;IAMxCC,GAAG,EAAE,EANmC;IAOxCC,IAAI,EAAE,EAPkC;IAQxCC,IAAI,EAAE,EARkC;IASxCC,IAAI,EAAE,EATkC;IAUxCC,KAAK,EAAE,EAViC;IAWxCC,IAAI,EAAE,EAXkC;IAYxCC,QAAQ,EAAE,EAZ8B;IAaxCC,SAAS,EAAE,EAb6B;IAcxCC,MAAM,EAAE,EAdgC;IAexCC,KAAK,EAAE,EAfiC;IAgBxCC,MAAM,EAAE;EAhBgC,CAA5C;;EAmBA,MAAMC,mBAAmB,GAAG,CACxBvC,IADwB,EAExBwC,IAFwB,EAGxBC,OAHwB,KAIjB;IACP,IAAIzC,IAAI,KAAK,KAAb,EAAoB;MAChB,MAAML,GAAG,GAAGD,MAAM,CAACD,IAAP,CAAY6B,aAAZ,EAA2BoB,IAA3B,CAAgC7C,GAAG,IAAI;QAC/C,MAAML,MAAM,GAAG8B,aAAa,CAACzB,GAAD,CAA5B;QACA,OAAOL,MAAM,CAACmD,QAAP,CAAgBH,IAAhB,CAAP;MACH,CAHW,CAAZ;;MAIA,IAAI,CAAC7C,GAAL,EAAU;QACN;MACH;;MACDiD,OAAO,CAACC,GAAR,CACK,4EADL;MAGAD,OAAO,CAACC,GAAR,CAAYjC,IAAI,CAACC,SAAL,CAAelB,GAAf,CAAZ;MACA,MAAM,IAAImD,cAAJ,CACD,mGADC,EAEF,sBAFE,EAGF;QACI9C,IADJ;QAEIwC;MAFJ,CAHE,CAAN;IAQH,CApBD,MAoBO,IAAIlB,aAAa,CAACtB,IAAD,CAAb,CAAoB2C,QAApB,CAA6BH,IAA7B,MAAuC,KAA3C,EAAkD;MACrD;IACH,CAFM,MAEA,IAAI,CAAAC,OAAO,SAAP,IAAAA,OAAO,WAAP,YAAAA,OAAO,CAAEM,QAAT,MAAsB,IAA1B,EAAgC;MACnC;IACH;;IACDH,OAAO,CAACC,GAAR,CAAa,0CAAyC7C,IAAK,KAAIwC,IAAK,EAApE;IACA,MAAM,IAAIM,cAAJ,CACD,4GADC,EAEF,sBAFE,EAGF;MACI9C,IADJ;MAEIwC;IAFJ,CAHE,CAAN;EAQH,CAvCD;;EAyCA,MAAMQ,eAAe,GAAG,CAAChD,IAAD,EAAoBwC,IAApB,KAA2C;IAC/D,IAAI,CAAClB,aAAa,CAACtB,IAAD,CAAlB,EAA0B;MACtB;IACH,CAFD,MAEO,IAAIsB,aAAa,CAACtB,IAAD,CAAb,CAAoB2C,QAApB,CAA6BH,IAA7B,CAAJ,EAAwC;MAC3C;IACH;;IACDlB,aAAa,CAACtB,IAAD,CAAb,CAAoBiD,IAApB,CAAyBT,IAAzB;EACH,CAPD;EAQA;AACJ;AACA;;;EACI,MAAMU,GAAG,GAAG,IAAAC,gBAAA,kCACJ9B,MAAM,CAACoB,OAAP,IAAkB,EADd,EAAZ;EAGA;AACJ;AACA;;EACIS,GAAG,CAACE,OAAJ,CAAY,SAAZ,EAAuBC,KAAK,IAAI;IAC5B,MAAMC,MAAM,GAAGD,KAAK,CAACC,MAArB;;IACA,IAAIrD,KAAK,CAACC,OAAN,CAAcoD,MAAd,CAAJ,EAA2B;MACvB,KAAK,MAAMC,CAAX,IAAgBD,MAAhB,EAAwB;QACpBN,eAAe,CAACO,CAAD,EAAIF,KAAK,CAACb,IAAV,CAAf;MACH;;MACD;IACH;;IACDQ,eAAe,CAACM,MAAD,EAASD,KAAK,CAACb,IAAf,CAAf;EACH,CATD;EAUA;AACJ;AACA;AACA;;EACI;AACJ;AACA;AACA;AACA;;EACIU,GAAG,CAACM,QAAJ,CAAaC,eAAb,EAA4B;IACxBC,YAAY,EAAE,EADU,CACP;;EADO,CAA5B;EAGA;AACJ;AACA;AACA;AACA;;EACIR,GAAG,CAACM,QAAJ,CAAaG,iBAAb,EAA8B;IAC1BC,MAAM,EAAE,IADkB;IAE1BC,SAAS,EAAE,IAFe;IAG1BC,qBAAqB,EAAE,CAACC,QAAD,EAAWC,CAAX,EAAcC,KAAd,KAAwB;MAC3CA,KAAK,CAACxD,IAAN,CAAW,GAAX;MACA,OAAQ,yBAAwBsD,QAAS,YAAzC;IACH,CANyB;IAO1BG,iBAAiB,EAAE;EAPO,CAA9B;EASA;AACJ;AACA;;EACI,MAAM1E,MAAqB,GAAG;IAC1B2E,OAAO,EAAE7C,aADiB;IAE1B8C,MAAM,EAAE,CAAC5B,IAAD,EAAO6B,OAAP,EAAgB5B,OAAhB,KAA4B;MAChCF,mBAAmB,CAAC,MAAD,EAASC,IAAT,EAAeC,OAAf,CAAnB;MACAS,GAAG,CAACoB,IAAJ,CAAS9B,IAAT,EAAe6B,OAAf;IACH,CALyB;IAM1BE,KAAK,EAAE,CAAC/B,IAAD,EAAO6B,OAAP,EAAgB5B,OAAhB,KAA4B;MAC/BF,mBAAmB,CAAC,KAAD,EAAQC,IAAR,EAAcC,OAAd,CAAnB;MACAS,GAAG,CAACsB,GAAJ,CAAQhC,IAAR,EAAc6B,OAAd;IACH,CATyB;IAU1BI,SAAS,EAAE,CAACjC,IAAD,EAAO6B,OAAP,EAAgB5B,OAAhB,KAA4B;MACnCF,mBAAmB,CAAC,SAAD,EAAYC,IAAZ,EAAkBC,OAAlB,CAAnB;MACAS,GAAG,CAACT,OAAJ,CAAYD,IAAZ,EAAkB6B,OAAlB;IACH,CAbyB;IAc1BK,QAAQ,EAAE,CAAClC,IAAD,EAAO6B,OAAP,EAAgB5B,OAAhB,KAA4B;MAClCF,mBAAmB,CAAC,QAAD,EAAWC,IAAX,EAAiBC,OAAjB,CAAnB;MACAS,GAAG,CAACyB,MAAJ,CAAWnC,IAAX,EAAiB6B,OAAjB;IACH,CAjByB;IAkB1BO,OAAO,EAAE,CAACpC,IAAD,EAAO6B,OAAP,EAAgB5B,OAAhB,KAA4B;MACjCF,mBAAmB,CAAC,OAAD,EAAUC,IAAV,EAAgBC,OAAhB,CAAnB;MACAS,GAAG,CAAC2B,KAAJ,CAAUrC,IAAV,EAAgB6B,OAAhB;IACH,CArByB;IAsB1BS,KAAK,EAAE,CAACtC,IAAD,EAAO6B,OAAP,EAAgB5B,OAAhB,KAA4B;MAC/BF,mBAAmB,CAAC,KAAD,EAAQC,IAAR,EAAcC,OAAd,CAAnB;MACAS,GAAG,CAAC6B,GAAJ,CAAQvC,IAAR,EAAc6B,OAAd;IACH,CAzByB;IA0B1BW,KAAK,EAAE,CAACxC,IAAD,EAAO6B,OAAP,EAAgB5B,OAAhB,KAA4B;MAC/BF,mBAAmB,CAAC,KAAD,EAAQC,IAAR,EAAcC,OAAd,CAAnB;MACAS,GAAG,CAACvD,GAAJ,CAAQ6C,IAAR,EAAc6B,OAAd;IACH,CA7ByB;IA8B1BY,MAAM,EAAE,CAACzC,IAAD,EAAO6B,OAAP,EAAgB5B,OAAhB,KAA4B;MAChCF,mBAAmB,CAAC,MAAD,EAASC,IAAT,EAAeC,OAAf,CAAnB;MACAS,GAAG,CAACgC,IAAJ,CAAS1C,IAAT,EAAe6B,OAAf;IACH;EAjCyB,CAA9B;EAmCA,IAAIc,OAAJ;;EACA,IAAI;IACAA,OAAO,GAAG,IAAIC,gBAAJ,CAAY;MAClBC,OAAO,EAAE;MACL;AAChB;AACA;AACA;MACgB,IAAAC,kCAAA,GALK,EAML,IAAIjE,MAAM,CAACgE,OAAP,IAAkB,EAAtB,CANK,CADS;;MASlB;AACZ;AACA;MACYE,cAAc,EAAEvE,OAAO,CAACC,GAAR,CAAYsE,cAZV;MAalBC,MAAM,EAAEtC,GAbU;MAclB1D;IAdkB,CAAZ,CAAV;EAgBH,CAjBD,CAiBE,OAAOiG,EAAP,EAAW;IACT7C,OAAO,CAACC,GAAR,CAAa,uCAAb;IACAD,OAAO,CAACC,GAAR,CAAYxC,cAAc,CAACoF,EAAD,CAA1B;IACA,MAAMA,EAAN;EACH;EAED;AACJ;AACA;;;EACIvC,GAAG,CAACwC,QAAJ,CAAa,QAAb,EAAuBP,OAAvB;EAEA;AACJ;AACA;AACA;AACA;AACA;AACA;;EACI,MAAMQ,YAAY,GAAGzC,GAAG,CAAC0C,MAAJ,CAAWP,OAAX,CAAmBQ,MAAnB,CAAuCC,wBAAA,CAAY9F,IAAnD,CAArB;EAEA;AACJ;AACA;;EACI,IAAI+F,eAAJ;;EACA,IAAI;IACA,KAAK,MAAMC,MAAX,IAAqBL,YAArB,EAAmC;MAC/BI,eAAe,GAAGC,MAAM,CAACzF,IAAzB;MACAyF,MAAM,CAACC,EAAP,6DACO/C,GAAG,CAAC0C,MAAJ,CAAWpG,MADlB;QAEI2F,OAAO,EAAEjC,GAAG,CAAC0C;MAFjB;IAIH;EACJ,CARD,CAQE,OAAOH,EAAP,EAAW;IACT7C,OAAO,CAACC,GAAR,CACK,yCACGkD,eAAe,GAAI,IAAGA,eAAgB,GAAvB,GAA4B,EAC9C,2DAHL;IAKAnD,OAAO,CAACC,GAAR,CAAYxC,cAAc,CAACoF,EAAD,CAA1B;IACA,MAAMA,EAAN;EACH;EAED;AACJ;AACA;AACA;;;EACIvC,GAAG,CAACE,OAAJ,CAAY,WAAZ,EAAyB,OAAO8C,OAAP,EAAgBjC,KAAhB,KAA0B;IAC/C;AACR;AACA;IACQ,MAAMkC,cAAc,GAAG5G,iBAAiB,CAAC+B,aAAD,CAAxC;IACA2C,KAAK,CAACmC,OAAN,CAAcD,cAAd;IACA;AACR;AACA;;IACQ,MAAMd,OAAO,GAAGnC,GAAG,CAAC0C,MAAJ,CAAWP,OAAX,CAAmBQ,MAAnB,CACZQ,8CAAA,CAAuBrG,IADX,CAAhB;IAIA,IAAIO,IAAJ;;IACA,IAAI;MACA,KAAK,MAAMyF,MAAX,IAAqBX,OAArB,EAA8B;QAC1B9E,IAAI,GAAGyF,MAAM,CAACzF,IAAd;QACA,MAAM+F,MAAM,GAAG,MAAMN,MAAM,CAACO,IAAP,CAAYL,OAAZ,EAAqBjC,KAArB,CAArB;;QACA,IAAIqC,MAAM,KAAK,KAAf,EAAsB;UAClB;QACH;MACJ;IACJ,CARD,CAQE,OAAOb,EAAP,EAAW;MACT7C,OAAO,CAACC,GAAR,CACK,oDACGtC,IAAI,GAAI,IAAGA,IAAK,GAAZ,GAAiB,EACxB,gCAHL;MAKAqC,OAAO,CAACC,GAAR,CAAYxC,cAAc,CAACoF,EAAD,CAA1B;MACA,MAAMA,EAAN;IACH;IACD;AACR;AACA;AACA;AACA;;;IACQ,IAAIS,OAAO,CAAC5C,MAAR,KAAmB,SAAvB,EAAkC;MAC9B;IACH;;IAED,IAAIW,KAAK,CAACuC,IAAV,EAAgB;MACZ;AACZ;AACA;MACY5D,OAAO,CAACC,GAAR,CACIjC,IAAI,CAACC,SAAL,CAAe;QACXL,OAAO,EAAG,wFADC;QAEXiG,WAAW,EACP;MAHO,CAAf,CADJ;MAOA;IACH;;IAEDxC,KAAK,CACAmC,OADL,6DACkBD,cADlB,GACqChF,eADrC,GAEKV,IAFL,CAEU,GAFV,EAGKiG,IAHL,CAGU,EAHV,EAIKC,MAJL;EAKH,CA3DD;EA6DAzD,GAAG,CAACE,OAAJ,CAAY,YAAZ,EAA0B,OAAO8C,OAAP,EAAgBjC,KAAhB,KAA0B;IAChDf,GAAG,CAAC0C,MAAJ,CAAWM,OAAX,GAAqBA,OAArB;IACAhD,GAAG,CAAC0C,MAAJ,CAAW3B,KAAX,GAAmBA,KAAnB;IACA,MAAMoB,OAAO,GAAGnC,GAAG,CAAC0C,MAAJ,CAAWP,OAAX,CAAmBQ,MAAnB,CAAyCe,kBAAA,CAAc5G,IAAvD,CAAhB;IACA,IAAIO,IAAJ;;IACA,IAAI;MACA,KAAK,MAAMyF,MAAX,IAAqBX,OAArB,EAA8B;QAC1B9E,IAAI,GAAGyF,MAAM,CAACzF,IAAd;QACA,MAAMyF,MAAM,CAACa,KAAP,CAAa3D,GAAG,CAAC0C,MAAjB,CAAN;MACH;IACJ,CALD,CAKE,OAAOH,EAAP,EAAW;MACT7C,OAAO,CAACC,GAAR,CACK,2CACGtC,IAAI,GAAI,IAAGA,IAAK,GAAZ,GAAiB,EACxB,iCAHL;MAKAqC,OAAO,CAACC,GAAR,CAAYxC,cAAc,CAACoF,EAAD,CAA1B;MACA,MAAMA,EAAN;IACH;EACJ,CAnBD;EAoBA;AACJ;AACA;;EACIvC,GAAG,CAACE,OAAJ,CAAY,YAAZ,EAA0B,YAAY;IAClC,MAAMiC,OAAO,GAAGnC,GAAG,CAAC0C,MAAJ,CAAWP,OAAX,CAAmBQ,MAAnB,CAA+CiB,wCAAA,CAAoB9G,IAAnE,CAAhB;IACA,IAAIO,IAAJ;;IACA,IAAI;MACA,KAAK,MAAMyF,MAAX,IAAqBX,OAArB,EAA8B;QAC1B9E,IAAI,GAAGyF,MAAM,CAACzF,IAAd;QACA,MAAMyF,MAAM,CAACa,KAAP,CAAa3D,GAAG,CAAC0C,MAAjB,CAAN;MACH;IACJ,CALD,CAKE,OAAOH,EAAP,EAAW;MACT7C,OAAO,CAACC,GAAR,CACK,iDACGtC,IAAI,GAAI,IAAGA,IAAK,GAAZ,GAAiB,EACxB,iCAHL;MAKAqC,OAAO,CAACC,GAAR,CAAYxC,cAAc,CAACoF,EAAD,CAA1B;MACA,MAAMA,EAAN;IACH;EACJ,CAjBD;EAmBA;AACJ;AACA;;EACI,MAAMsB,gBAAuD,GAAG,OAAO/C,CAAP,EAAUgD,EAAV,EAAcC,OAAd,KAA0B;IACtF,MAAM5B,OAAO,GAAGnC,GAAG,CAAC0C,MAAJ,CAAWP,OAAX,CAAmBQ,MAAnB,CAA+CqB,wCAAA,CAAoBlH,IAAnE,CAAhB;IACA,IAAIO,IAAJ;;IACA,IAAI;MACA,KAAK,MAAMyF,MAAX,IAAqBX,OAArB,EAA8B;QAC1B9E,IAAI,GAAGyF,MAAM,CAACzF,IAAd;QACA,MAAMyF,MAAM,CAACmB,MAAP,CAAcjE,GAAG,CAAC0C,MAAlB,EAA0BqB,OAA1B,CAAN;MACH;IACJ,CALD,CAKE,OAAOxB,EAAP,EAAW;MACT7C,OAAO,CAACC,GAAR,CACK,iDACGtC,IAAI,GAAI,IAAGA,IAAK,GAAZ,GAAiB,EACxB,uCAHL;MAKAqC,OAAO,CAACC,GAAR,CAAYxC,cAAc,CAACoF,EAAD,CAA1B;MACA,MAAMA,EAAN;IACH;;IACD,OAAOwB,OAAP;EACH,CAlBD;;EAoBA/D,GAAG,CAACE,OAAJ,CAAY,kBAAZ,EAAgC2D,gBAAhC;EAEA7D,GAAG,CAACkE,eAAJ,CAAiC,OAAO9G,KAAP,EAAc4F,OAAd,EAAuBjC,KAAvB,KAAiC;IAC9D,OAAOA,KAAK,CACPoD,MADE,CACK,GADL,EAEFjB,OAFE,CAEM;MACL,iBAAiB;IADZ,CAFN,EAKFM,IALE;IAMC;AAChB;AACA;IACgB9F,IAAI,CAACC,SAAL,CAAe;MACXL,OAAO,EAAEF,KAAK,CAACE,OADJ;MAEXC,IAAI,EAAEH,KAAK,CAACG,IAFD;MAGXE,IAAI,EAAEL,KAAK,CAACK;IAHD,CAAf,CATD,CAAP;EAeH,CAhBD;EAkBAuC,GAAG,CAACE,OAAJ,CAAY,SAAZ,EAAuB,OAAOY,CAAP,EAAUC,KAAV,EAAiB3D,KAAjB,KAAgC;IACnD,MAAM+E,OAAO,GAAGnC,GAAG,CAAC0C,MAAJ,CAAWP,OAAX,CAAmBQ,MAAnB,CAA8CyB,sCAAA,CAAmBtH,IAAjE,CAAhB;IACA;AACR;AACA;;IACQ4C,OAAO,CAACC,GAAR,CAAY,iBAAZ;IACAD,OAAO,CAACC,GAAR,CAAYxC,cAAc,CAACC,KAAD,CAA1B;IAEA2D,KAAK,CACAoD,MADL,CACY,GADZ,EAEKjB,OAFL,CAEa;MACL,iBAAiB;IADZ,CAFb,EAKKM,IALL;IAMQ;AAChB;AACA;IACgB9F,IAAI,CAACC,SAAL,CAAe;MACXL,OAAO,EAAEF,KAAK,CAACE,OADJ;MAEXC,IAAI,EAAEH,KAAK,CAACG,IAFD;MAGXE,IAAI,EAAEL,KAAK,CAACK;IAHD,CAAf,CATR;IAgBA,MAAM0D,OAAO,GAAG,IAAAkD,sBAAA,EACZlC,OAAO,CAACmC,GAAR,CAAYC,EAAE,IAAI;MACd,OAAO,CAACtC,OAAD,EAAmB7E,KAAnB,EAAiCoH,IAAjC,KAAoD;QACvD,OAAOD,EAAE,CAACN,MAAH,CAAUhC,OAAV,EAAmB7E,KAAnB,EAA0BoH,IAA1B,CAAP;MACH,CAFD;IAGH,CAJD,CADY,CAAhB;IAOA,MAAMrD,OAAO,CAACnB,GAAG,CAAC0C,MAAL,EAAatF,KAAb,CAAb;IAEA,OAAO2D,KAAP;EACH,CAlCD;EAoCA;AACJ;AACA;;EACI,MAAM0D,aAAa,GAAGzE,GAAG,CAAC0C,MAAJ,CAAWP,OAAX,CAAmBQ,MAAnB,CAA+C+B,wCAAA,CAAoB5H,IAAnE,CAAtB;EAEA,IAAI6H,uBAAJ;;EACA,IAAI;IACA,KAAK,MAAM7B,MAAX,IAAqB2B,aAArB,EAAoC;MAChCE,uBAAuB,GAAG7B,MAAM,CAACzF,IAAjC;MACAyF,MAAM,CAAC8B,MAAP,CAAc5E,GAAd;IACH;EACJ,CALD,CAKE,OAAOuC,EAAP,EAAW;IACT7C,OAAO,CAACC,GAAR,CACK,iDACGgF,uBAAuB,GAAI,IAAGA,uBAAwB,GAA/B,GAAoC,EAC9D,qDAHL;IAKAjF,OAAO,CAACC,GAAR,CAAYxC,cAAc,CAACoF,EAAD,CAA1B;IACA,MAAMA,EAAN;EACH;;EAED,OAAOvC,GAAP;AACH,CApaM"}