@webiny/handler 0.0.0-unstable.c2780f51fe → 0.0.0-unstable.c27f4d8a31

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 (89) hide show
  1. package/Context.d.ts +4 -5
  2. package/Context.js +11 -25
  3. package/Context.js.map +1 -1
  4. package/PreHandler/IPreHandler.d.ts +9 -0
  5. package/PreHandler/IPreHandler.js +7 -0
  6. package/PreHandler/IPreHandler.js.map +1 -0
  7. package/PreHandler/IfNotOptionsRequest.d.ts +9 -0
  8. package/PreHandler/IfNotOptionsRequest.js +21 -0
  9. package/PreHandler/IfNotOptionsRequest.js.map +1 -0
  10. package/PreHandler/IfOptionsRequest.d.ts +9 -0
  11. package/PreHandler/IfOptionsRequest.js +21 -0
  12. package/PreHandler/IfOptionsRequest.js.map +1 -0
  13. package/PreHandler/PreHandler.d.ts +9 -0
  14. package/PreHandler/PreHandler.js +17 -0
  15. package/PreHandler/PreHandler.js.map +1 -0
  16. package/PreHandler/ProcessBeforeHandlerPlugins.d.ts +10 -0
  17. package/PreHandler/ProcessBeforeHandlerPlugins.js +24 -0
  18. package/PreHandler/ProcessBeforeHandlerPlugins.js.map +1 -0
  19. package/PreHandler/ProcessContextPlugins.d.ts +10 -0
  20. package/PreHandler/ProcessContextPlugins.js +24 -0
  21. package/PreHandler/ProcessContextPlugins.js.map +1 -0
  22. package/PreHandler/ProcessHandlerOnRequestPlugins.d.ts +10 -0
  23. package/PreHandler/ProcessHandlerOnRequestPlugins.js +26 -0
  24. package/PreHandler/ProcessHandlerOnRequestPlugins.js.map +1 -0
  25. package/PreHandler/SendEarlyOptionsResponse.d.ts +9 -0
  26. package/PreHandler/SendEarlyOptionsResponse.js +31 -0
  27. package/PreHandler/SendEarlyOptionsResponse.js.map +1 -0
  28. package/PreHandler/SetDefaultHeaders.d.ts +9 -0
  29. package/PreHandler/SetDefaultHeaders.js +57 -0
  30. package/PreHandler/SetDefaultHeaders.js.map +1 -0
  31. package/README.md +10 -14
  32. package/ResponseHeaders.d.ts +24 -0
  33. package/ResponseHeaders.js +39 -0
  34. package/ResponseHeaders.js.map +1 -0
  35. package/abstractions/Reply.d.ts +5 -0
  36. package/abstractions/Reply.js +4 -0
  37. package/abstractions/Reply.js.map +1 -0
  38. package/abstractions/Request.d.ts +5 -0
  39. package/abstractions/Request.js +4 -0
  40. package/abstractions/Request.js.map +1 -0
  41. package/fastify.d.ts +8 -5
  42. package/fastify.js +223 -207
  43. package/fastify.js.map +1 -1
  44. package/index.d.ts +17 -9
  45. package/index.js +19 -103
  46. package/index.js.map +1 -1
  47. package/package.json +16 -26
  48. package/plugins/BeforeHandlerPlugin.d.ts +1 -1
  49. package/plugins/BeforeHandlerPlugin.js +6 -15
  50. package/plugins/BeforeHandlerPlugin.js.map +1 -1
  51. package/plugins/EventPlugin.d.ts +3 -3
  52. package/plugins/EventPlugin.js +6 -16
  53. package/plugins/EventPlugin.js.map +1 -1
  54. package/plugins/HandlerErrorPlugin.d.ts +6 -3
  55. package/plugins/HandlerErrorPlugin.js +6 -15
  56. package/plugins/HandlerErrorPlugin.js.map +1 -1
  57. package/plugins/HandlerOnRequestPlugin.d.ts +9 -8
  58. package/plugins/HandlerOnRequestPlugin.js +15 -16
  59. package/plugins/HandlerOnRequestPlugin.js.map +1 -1
  60. package/plugins/HandlerResultPlugin.d.ts +1 -1
  61. package/plugins/HandlerResultPlugin.js +6 -15
  62. package/plugins/HandlerResultPlugin.js.map +1 -1
  63. package/plugins/ModifyFastifyPlugin.d.ts +2 -2
  64. package/plugins/ModifyFastifyPlugin.js +6 -15
  65. package/plugins/ModifyFastifyPlugin.js.map +1 -1
  66. package/plugins/ModifyResponseHeadersPlugin.d.ts +14 -0
  67. package/plugins/ModifyResponseHeadersPlugin.js +16 -0
  68. package/plugins/ModifyResponseHeadersPlugin.js.map +1 -0
  69. package/plugins/OnRequestResponseSendPlugin.d.ts +26 -0
  70. package/plugins/OnRequestResponseSendPlugin.js +31 -0
  71. package/plugins/OnRequestResponseSendPlugin.js.map +1 -0
  72. package/plugins/OnRequestTimeoutPlugin.d.ts +12 -0
  73. package/plugins/OnRequestTimeoutPlugin.js +16 -0
  74. package/plugins/OnRequestTimeoutPlugin.js.map +1 -0
  75. package/plugins/RoutePlugin.d.ts +2 -2
  76. package/plugins/RoutePlugin.js +6 -15
  77. package/plugins/RoutePlugin.js.map +1 -1
  78. package/stringifyError.d.ts +5 -0
  79. package/stringifyError.js +20 -0
  80. package/stringifyError.js.map +1 -0
  81. package/suppressPunycodeWarnings.d.ts +1 -0
  82. package/suppressPunycodeWarnings.js +10 -0
  83. package/suppressPunycodeWarnings.js.map +1 -0
  84. package/types.d.ts +10 -13
  85. package/types.js +2 -17
  86. package/types.js.map +1 -1
  87. package/middleware.d.ts +0 -4
  88. package/middleware.js +0 -45
  89. package/middleware.js.map +0 -1
@@ -0,0 +1,4 @@
1
+ import { createAbstraction } from "@webiny/feature/api";
2
+ export const Request = createAbstraction("Request");
3
+
4
+ //# sourceMappingURL=Request.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"names":["createAbstraction","Request"],"sources":["Request.ts"],"sourcesContent":["import { createAbstraction } from \"@webiny/feature/api\";\nimport type { Request as IRequest } from \"~/types.js\";\n\nexport const Request = createAbstraction<IRequest>(\"Request\");\n\nexport namespace Request {\n export type Interface = IRequest;\n}\n"],"mappings":"AAAA,SAASA,iBAAiB,QAAQ,qBAAqB;AAGvD,OAAO,MAAMC,OAAO,GAAGD,iBAAiB,CAAW,SAAS,CAAC","ignoreList":[]}
package/fastify.d.ts CHANGED
@@ -1,8 +1,11 @@
1
- /// <reference types="node" />
2
- import { PluginCollection } from "@webiny/plugins/types";
3
- import { FastifyServerOptions as ServerOptions } from "fastify";
1
+ import type { PluginCollection } from "@webiny/plugins/types.js";
2
+ import { PluginsContainer } from "@webiny/plugins/types.js";
3
+ import type { FastifyInstance, FastifyServerOptions as ServerOptions } from "fastify";
4
4
  export interface CreateHandlerParams {
5
- plugins: PluginCollection;
5
+ plugins: PluginCollection | PluginsContainer;
6
6
  options?: ServerOptions;
7
+ debug?: boolean;
7
8
  }
8
- export declare const createHandler: (params: CreateHandlerParams) => import("fastify").FastifyInstance<import("fastify").RawServerDefault, import("http").IncomingMessage, import("http").ServerResponse<import("http").IncomingMessage>, import("fastify").FastifyBaseLogger, import("fastify").FastifyTypeProviderDefault> & PromiseLike<import("fastify").FastifyInstance<import("fastify").RawServerDefault, import("http").IncomingMessage, import("http").ServerResponse<import("http").IncomingMessage>, import("fastify").FastifyBaseLogger, import("fastify").FastifyTypeProviderDefault>>;
9
+ export declare const createHandler: (params: CreateHandlerParams) => FastifyInstance<import("fastify").RawServerDefault, import("http").IncomingMessage, import("http").ServerResponse<import("http").IncomingMessage>, import("fastify").FastifyBaseLogger, import("fastify").FastifyTypeProviderDefault> & PromiseLike<FastifyInstance<import("fastify").RawServerDefault, import("http").IncomingMessage, import("http").ServerResponse<import("http").IncomingMessage>, import("fastify").FastifyBaseLogger, import("fastify").FastifyTypeProviderDefault>> & {
10
+ __linterBrands: "SafePromiseLike";
11
+ };
package/fastify.js CHANGED
@@ -1,76 +1,48 @@
1
- "use strict";
2
-
3
- var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault").default;
4
- Object.defineProperty(exports, "__esModule", {
5
- value: true
6
- });
7
- exports.createHandler = void 0;
8
- var _objectSpread2 = _interopRequireDefault(require("@babel/runtime/helpers/objectSpread2"));
9
- var _fastify = _interopRequireDefault(require("fastify"));
10
- var _utils = require("@webiny/utils");
11
- var _Context = require("./Context");
12
- var _error = _interopRequireDefault(require("@webiny/error"));
13
- var _RoutePlugin = require("./plugins/RoutePlugin");
14
- var _handlerClient = require("@webiny/handler-client");
15
- var _cookie = _interopRequireDefault(require("@fastify/cookie"));
16
- var _compress = _interopRequireDefault(require("@fastify/compress"));
17
- var _middleware = require("./middleware");
18
- var _api = require("@webiny/api");
19
- var _BeforeHandlerPlugin = require("./plugins/BeforeHandlerPlugin");
20
- var _HandlerResultPlugin = require("./plugins/HandlerResultPlugin");
21
- var _HandlerErrorPlugin = require("./plugins/HandlerErrorPlugin");
22
- var _ModifyFastifyPlugin = require("./plugins/ModifyFastifyPlugin");
23
- var _HandlerOnRequestPlugin = require("./plugins/HandlerOnRequestPlugin");
24
- const DEFAULT_HEADERS = (0, _objectSpread2.default)({
25
- "Cache-Control": "no-store",
26
- "Content-Type": "application/json; charset=utf-8",
27
- "Access-Control-Allow-Origin": "*",
28
- "Access-Control-Allow-Headers": "*",
29
- "Access-Control-Allow-Methods": "OPTIONS,POST,GET,DELETE,PUT,PATCH"
30
- }, (0, _utils.getWebinyVersionHeaders)());
31
- const getDefaultHeaders = routes => {
32
- /**
33
- * If we are accepting all headers, just output that one.
34
- */
35
- const keys = Object.keys(routes);
36
- const all = keys.every(key => routes[key].length > 0);
37
- if (all) {
38
- return (0, _objectSpread2.default)((0, _objectSpread2.default)({}, DEFAULT_HEADERS), {}, {
39
- "Access-Control-Allow-Methods": "*"
40
- });
41
- }
42
- return (0, _objectSpread2.default)((0, _objectSpread2.default)({}, DEFAULT_HEADERS), {}, {
43
- "Access-Control-Allow-Methods": keys.filter(type => {
44
- if (!routes[type] || Array.isArray(routes[type]) === false) {
45
- return false;
46
- }
47
- return routes[type].length > 0;
48
- }).sort().join(",")
1
+ import { PluginsContainer } from "@webiny/plugins/types.js";
2
+ import fastify from "fastify";
3
+ import { middleware } from "@webiny/utils";
4
+ import { Context } from "./Context.js";
5
+ import WebinyError from "@webiny/error";
6
+ import { RoutePlugin } from "./plugins/RoutePlugin.js";
7
+ import { createHandlerClient } from "@webiny/handler-client";
8
+ import fastifyCookie from "@fastify/cookie";
9
+ import fastifyCompress from "@fastify/compress";
10
+ import { ContextPlugin } from "@webiny/api";
11
+ import { BeforeHandlerPlugin } from "./plugins/BeforeHandlerPlugin.js";
12
+ import { HandlerResultPlugin } from "./plugins/HandlerResultPlugin.js";
13
+ import { HandlerErrorPlugin } from "./plugins/HandlerErrorPlugin.js";
14
+ import { ModifyFastifyPlugin } from "./plugins/ModifyFastifyPlugin.js";
15
+ import { HandlerOnRequestPlugin } from "./plugins/HandlerOnRequestPlugin.js";
16
+ import { ResponseHeaders } from "./ResponseHeaders.js";
17
+ import { ModifyResponseHeadersPlugin } from "./plugins/ModifyResponseHeadersPlugin.js";
18
+ import { SetDefaultHeaders } from "./PreHandler/SetDefaultHeaders.js";
19
+ import { PreHandler } from "./PreHandler/PreHandler.js";
20
+ import { stringifyError } from "./stringifyError.js";
21
+ import { ProcessHandlerOnRequestPlugins } from "./PreHandler/ProcessHandlerOnRequestPlugins.js";
22
+ import { ProcessContextPlugins } from "./PreHandler/ProcessContextPlugins.js";
23
+ import { IfNotOptionsRequest } from "./PreHandler/IfNotOptionsRequest.js";
24
+ import { ProcessBeforeHandlerPlugins } from "./PreHandler/ProcessBeforeHandlerPlugins.js";
25
+ import { IfOptionsRequest } from "./PreHandler/IfOptionsRequest.js";
26
+ import { SendEarlyOptionsResponse } from "./PreHandler/SendEarlyOptionsResponse.js";
27
+ import { OnRequestTimeoutPlugin } from "./plugins/OnRequestTimeoutPlugin.js";
28
+ import { OnRequestResponseSendPlugin } from "./plugins/OnRequestResponseSendPlugin.js";
29
+ import { Request } from "./abstractions/Request.js";
30
+ import { Reply } from "./abstractions/Reply.js";
31
+ const modifyResponseHeaders = (app, request, reply) => {
32
+ const modifyHeaders = app.webiny.plugins.byType(ModifyResponseHeadersPlugin.type);
33
+ const replyHeaders = reply.getHeaders();
34
+ const headers = ResponseHeaders.create(replyHeaders);
35
+ modifyHeaders.forEach(plugin => {
36
+ plugin.modify(request, headers);
49
37
  });
38
+
39
+ // Exclude 'set-cookie' header to avoid duplication.
40
+ // Cookies are managed by @fastify/cookie and calling reply.headers() with 'set-cookie' duplicates them.
41
+ const headersToSet = headers.getHeaders();
42
+ delete headersToSet["set-cookie"];
43
+ reply.headers(headersToSet);
50
44
  };
51
- const stringifyError = error => {
52
- var _error$constructor;
53
- const {
54
- name,
55
- message,
56
- code,
57
- stack,
58
- data
59
- } = error;
60
- return JSON.stringify((0, _objectSpread2.default)((0, _objectSpread2.default)({}, error), {}, {
61
- constructorName: ((_error$constructor = error.constructor) === null || _error$constructor === void 0 ? void 0 : _error$constructor.name) || "UnknownError",
62
- name: name || "No error name",
63
- message: message || "No error message",
64
- code: code || "NO_CODE",
65
- data,
66
- stack: process.env.DEBUG === "true" ? stack : "Turn on the debug flag to see the stack."
67
- }));
68
- };
69
- const OPTIONS_HEADERS = {
70
- "Access-Control-Max-Age": "86400",
71
- "Cache-Control": "public, max-age=86400"
72
- };
73
- const createHandler = params => {
45
+ export const createHandler = params => {
74
46
  const definedRoutes = {
75
47
  POST: [],
76
48
  GET: [],
@@ -87,11 +59,14 @@ const createHandler = params => {
87
59
  PROPPATCH: [],
88
60
  SEARCH: [],
89
61
  TRACE: [],
90
- UNLOCK: []
62
+ UNLOCK: [],
63
+ REPORT: [],
64
+ MKCALENDAR: []
91
65
  };
92
66
  const throwOnDefinedRoute = (type, path, options) => {
93
67
  if (type === "ALL") {
94
- const all = Object.keys(definedRoutes).find(key => {
68
+ const all = Object.keys(definedRoutes).find(k => {
69
+ const key = k.toUpperCase();
95
70
  const routes = definedRoutes[key];
96
71
  return routes.includes(path);
97
72
  });
@@ -100,22 +75,23 @@ const createHandler = params => {
100
75
  }
101
76
  console.error(`Error while registering onAll route. One of the routes is already defined.`);
102
77
  console.error(JSON.stringify(all));
103
- throw new _error.default(`You cannot override a route with onAll() method, please remove unnecessary route from the system.`, "OVERRIDE_ROUTE_ERROR", {
78
+ throw new WebinyError(`You cannot override a route with onAll() method, please remove unnecessary route from the system.`, "OVERRIDE_ROUTE_ERROR", {
104
79
  type,
105
80
  path
106
81
  });
107
82
  } else if (definedRoutes[type].includes(path) === false) {
108
83
  return;
109
- } else if ((options === null || options === void 0 ? void 0 : options.override) === true) {
84
+ } else if (options?.override === true) {
110
85
  return;
111
86
  }
112
87
  console.error(`Error while trying to override route: [${type}] ${path}`);
113
- throw new _error.default(`When you are trying to override existing route, you must send "override" parameter when adding that route.`, "OVERRIDE_ROUTE_ERROR", {
88
+ throw new WebinyError(`When you are trying to override existing route, you must send "override" parameter when adding that route.`, "OVERRIDE_ROUTE_ERROR", {
114
89
  type,
115
90
  path
116
91
  });
117
92
  };
118
- const addDefinedRoute = (type, path) => {
93
+ const addDefinedRoute = (input, path) => {
94
+ const type = input.toUpperCase();
119
95
  if (!definedRoutes[type]) {
120
96
  return;
121
97
  } else if (definedRoutes[type].includes(path)) {
@@ -123,12 +99,20 @@ const createHandler = params => {
123
99
  }
124
100
  definedRoutes[type].push(path);
125
101
  };
102
+
126
103
  /**
127
104
  * We must attach the server to our internal context if we want to have it accessible.
128
105
  */
129
- const app = (0, _fastify.default)((0, _objectSpread2.default)({}, params.options || {}));
106
+ const app = fastify({
107
+ bodyLimit: 536870912,
108
+ // 512MB
109
+ disableRequestLogging: true,
110
+ allowErrorHandlerOverride: true,
111
+ ...(params.options || {})
112
+ });
113
+
130
114
  /**
131
- * We need to register routes in our system so we can output headers later on and dissallow overriding routes.
115
+ * We need to register routes in our system to output headers later on, and disallow route overriding.
132
116
  */
133
117
  app.addHook("onRoute", route => {
134
118
  const method = route.method;
@@ -149,7 +133,7 @@ const createHandler = params => {
149
133
  *
150
134
  * https://github.com/fastify/fastify-cookie
151
135
  */
152
- app.register(_cookie.default, {
136
+ app.register(fastifyCookie, {
153
137
  parseOptions: {} // options for parsing cookies
154
138
  });
155
139
  /**
@@ -157,7 +141,7 @@ const createHandler = params => {
157
141
  *
158
142
  * https://github.com/fastify/fastify-compress
159
143
  */
160
- app.register(_compress.default, {
144
+ app.register(fastifyCompress, {
161
145
  global: true,
162
146
  threshold: 1024,
163
147
  onUnsupportedEncoding: (encoding, _, reply) => {
@@ -205,19 +189,20 @@ const createHandler = params => {
205
189
  }
206
190
  };
207
191
  let context;
192
+ const plugins = new PluginsContainer([
193
+ /**
194
+ * We must have handlerClient by default.
195
+ * And it must be one of the first context plugins applied.
196
+ */
197
+ createHandlerClient()]);
198
+ plugins.merge(params.plugins || []);
208
199
  try {
209
- context = new _Context.Context({
210
- plugins: [
200
+ context = new Context({
201
+ plugins,
211
202
  /**
212
- * We must have handlerClient by default.
213
- * And it must be one of the first context plugins applied.
214
- */
215
- (0, _handlerClient.createHandlerClient)(), ...(params.plugins || [])],
216
- /**
217
- * Inserted via webpack on build time.
203
+ * Inserted via webpack at build time.
218
204
  */
219
205
  WEBINY_VERSION: process.env.WEBINY_VERSION,
220
- server: app,
221
206
  routes
222
207
  });
223
208
  } catch (ex) {
@@ -232,118 +217,68 @@ const createHandler = params => {
232
217
  app.decorate("webiny", context);
233
218
 
234
219
  /**
235
- * We have few types of triggers:
236
- * * Events - EventPlugin
237
- * * Routes - RoutePlugin
238
- *
239
- * Routes are registered in fastify but events must be handled in package which implements cloud specific methods.
220
+ * To prevent Unsupported Media Type errors on OPTIONS requests with a body,
221
+ * we need to have a custom parser
240
222
  */
241
- const routePlugins = app.webiny.plugins.byType(_RoutePlugin.RoutePlugin.type);
242
-
243
- /**
244
- * Add routes to the system.
245
- */
246
- let routePluginName;
247
- try {
248
- for (const plugin of routePlugins) {
249
- routePluginName = plugin.name;
250
- plugin.cb((0, _objectSpread2.default)((0, _objectSpread2.default)({}, app.webiny.routes), {}, {
251
- context: app.webiny
252
- }));
253
- }
254
- } catch (ex) {
255
- console.error(`Error while running the "RoutePlugin" ${routePluginName ? `(${routePluginName})` : ""} plugin in the beginning of the "createHandler" callable.`);
256
- console.error(stringifyError(ex));
257
- throw ex;
258
- }
259
-
260
- /**
261
- * On every request we add default headers, which can be changed later.
262
- * Also, if it is an options request, we skip everything after this hook and output options headers.
263
- */
264
- app.addHook("onRequest", async (request, reply) => {
265
- /**
266
- * Our default headers are always set. Users can override them.
267
- */
268
- const defaultHeaders = getDefaultHeaders(definedRoutes);
269
- reply.headers(defaultHeaders);
270
- /**
271
- * Users can define their own custom handlers for the onRequest event - so let's run them first.
272
- */
273
- const plugins = app.webiny.plugins.byType(_HandlerOnRequestPlugin.HandlerOnRequestPlugin.type);
274
- let name;
275
- try {
276
- for (const plugin of plugins) {
277
- name = plugin.name;
278
- const result = await plugin.exec(request, reply);
279
- if (result === false) {
280
- return;
281
- }
282
- }
283
- } catch (ex) {
284
- console.error(`Error while running the "HandlerOnRequestPlugin" ${name ? `(${name})` : ""} plugin in the onRequest hook.`);
285
- console.error(stringifyError(ex));
286
- throw ex;
287
- }
288
- /**
289
- * 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.
290
- *
291
- * Users can prevent this by creating their own HandlerOnRequestPlugin and returning false as the result of the callable.
292
- */
293
- if (request.method !== "OPTIONS") {
223
+ app.addContentTypeParser("application/json", {
224
+ parseAs: "string",
225
+ bodyLimit: 1024 * 1024
226
+ }, (req, body, done) => {
227
+ if (req.method === "OPTIONS") {
228
+ done(null, undefined);
294
229
  return;
295
230
  }
296
- if (reply.sent) {
297
- /**
298
- * At this point throwing an exception will not do anything with the response. So just log it.
299
- */
300
- console.error(JSON.stringify({
301
- message: `Output was already sent. Please check custom plugins of type "HandlerOnRequestPlugin".`,
302
- explanation: "This error can happen if the user plugin ended the reply, but did not return false as response."
303
- }));
304
- return;
305
- }
306
- reply.headers((0, _objectSpread2.default)((0, _objectSpread2.default)({}, defaultHeaders), OPTIONS_HEADERS)).code(204).send("").hijack();
307
- });
308
- app.addHook("preParsing", async (request, reply) => {
309
- app.webiny.request = request;
310
- app.webiny.reply = reply;
311
- const plugins = app.webiny.plugins.byType(_api.ContextPlugin.type);
312
- let name;
313
231
  try {
314
- for (const plugin of plugins) {
315
- name = plugin.name;
316
- await plugin.apply(app.webiny);
317
- }
318
- } catch (ex) {
319
- console.error(`Error while running the "ContextPlugin" ${name ? `(${name})` : ""} plugin in the preParsing hook.`);
320
- console.error(stringifyError(ex));
321
- throw ex;
232
+ const json = typeof body === "string" ? body : body.toString("utf8");
233
+ done(null, JSON.parse(json));
234
+ } catch (err) {
235
+ done(err);
322
236
  }
323
237
  });
238
+
324
239
  /**
240
+ * With this we ensure that an undefined request body is not parsed on OPTIONS requests,
241
+ * in case there's a `content-type` header set for whatever reason.
325
242
  *
243
+ * @see https://fastify.dev/docs/latest/Reference/ContentTypeParser/#content-type-parser
326
244
  */
327
- app.addHook("preHandler", async () => {
328
- const plugins = app.webiny.plugins.byType(_BeforeHandlerPlugin.BeforeHandlerPlugin.type);
329
- let name;
330
- try {
331
- for (const plugin of plugins) {
332
- name = plugin.name;
333
- await plugin.apply(app.webiny);
334
- }
335
- } catch (ex) {
336
- console.error(`Error while running the "BeforeHandlerPlugin" ${name ? `(${name})` : ""} plugin in the preHandler hook.`);
337
- console.error(stringifyError(ex));
338
- throw ex;
245
+ app.addHook("onRequest", async request => {
246
+ if (request.method === "OPTIONS" && request.body === undefined) {
247
+ request.headers["content-type"] = undefined;
339
248
  }
340
249
  });
341
250
 
342
251
  /**
343
- *
252
+ * At this point, request body is properly parsed, and we can execute Webiny business logic.
253
+ * - set default headers
254
+ * - process `HandlerOnRequestPlugin`
255
+ * - if OPTIONS request, exit early
256
+ * - process `ContextPlugin`
257
+ * - process `BeforeHandlerPlugin`
344
258
  */
345
- const preSerialization = async (_, __, payload) => {
346
- const plugins = app.webiny.plugins.byType(_HandlerResultPlugin.HandlerResultPlugin.type);
259
+ app.addHook("preHandler", async (request, reply) => {
260
+ app.webiny.request = request;
261
+ app.webiny.reply = reply;
262
+
263
+ // Bind request and reply to DI container for runtime access
264
+ if (app.webiny.container) {
265
+ app.webiny.container.registerInstance(Request, request);
266
+ app.webiny.container.registerInstance(Reply, reply);
267
+ }
268
+ /**
269
+ * Default code to 200 - so we do not need to set it again.
270
+ * Usually we set errors manually when we use reply.send.
271
+ */
272
+ reply.code(200);
273
+ const handlerOnRequestPlugins = app.webiny.plugins.byType(HandlerOnRequestPlugin.type);
274
+ const contextPlugins = app.webiny.plugins.byType(ContextPlugin.type);
275
+ const beforeHandlerPlugins = app.webiny.plugins.byType(BeforeHandlerPlugin.type);
276
+ const modifyHeadersPlugins = app.webiny.plugins.byType(ModifyResponseHeadersPlugin.type);
277
+ const preHandler = new PreHandler([new SetDefaultHeaders(definedRoutes), new ProcessHandlerOnRequestPlugins(handlerOnRequestPlugins), new IfNotOptionsRequest([new ProcessContextPlugins(app.webiny, contextPlugins), new ProcessBeforeHandlerPlugins(app.webiny, beforeHandlerPlugins)]), new IfOptionsRequest([new SendEarlyOptionsResponse(modifyHeadersPlugins)])]);
278
+ await preHandler.execute(request, reply, app.webiny);
279
+ });
280
+ app.addHook("preSerialization", async (_, __, payload) => {
281
+ const plugins = app.webiny.plugins.byType(HandlerResultPlugin.type);
347
282
  let name;
348
283
  try {
349
284
  for (const plugin of plugins) {
@@ -356,9 +291,31 @@ const createHandler = params => {
356
291
  throw ex;
357
292
  }
358
293
  return payload;
359
- };
360
- app.addHook("preSerialization", preSerialization);
361
- app.setErrorHandler(async (error, request, reply) => {
294
+ });
295
+ app.setErrorHandler(async (error, _, reply) => {
296
+ /**
297
+ * IMPORTANT! Do not send anything if reply was already sent.
298
+ */
299
+ if (reply.sent) {
300
+ console.warn("Reply already sent, cannot send the result (handler:setErrorHandler).");
301
+ return reply;
302
+ }
303
+ if (error.code?.startsWith("Authentication/")) {
304
+ return reply.status(401).headers({
305
+ "Cache-Control": "no-store"
306
+ }).send(JSON.stringify({
307
+ message: error.message,
308
+ code: error.code
309
+ }));
310
+ }
311
+ if (error.code === "Tenancy/TenantDisabled") {
312
+ return reply.status(503).headers({
313
+ "Cache-Control": "no-store"
314
+ }).send(JSON.stringify({
315
+ message: error.message,
316
+ code: error.code
317
+ }));
318
+ }
362
319
  return reply.status(500).headers({
363
320
  "Cache-Control": "no-store"
364
321
  }).send(
@@ -372,24 +329,37 @@ const createHandler = params => {
372
329
  }));
373
330
  });
374
331
  app.addHook("onError", async (_, reply, error) => {
375
- const plugins = app.webiny.plugins.byType(_HandlerErrorPlugin.HandlerErrorPlugin.type);
332
+ const plugins = app.webiny.plugins.byType(HandlerErrorPlugin.type);
376
333
  /**
377
334
  * Log error to cloud, as these can be extremely annoying to debug!
378
335
  */
379
- console.error("@webiny/handler");
380
- console.error(stringifyError(error));
381
- reply.status(500).headers({
382
- "Cache-Control": "no-store"
383
- }).send(
336
+ console.error("Logging error in @webiny/handler");
337
+ try {
338
+ console.error(stringifyError(error));
339
+ } catch (ex) {
340
+ console.warn("Could not stringify error:");
341
+ console.log(error);
342
+ console.error("Stringify error:", ex);
343
+ }
384
344
  /**
385
- * When we are sending the error in the response, we cannot send the whole error object, as it might contain some sensitive data.
345
+ * IMPORTANT! Do not send anything if reply was already sent.
386
346
  */
387
- JSON.stringify({
388
- message: error.message,
389
- code: error.code,
390
- data: error.data
391
- }));
392
- const handler = (0, _middleware.middleware)(plugins.map(pl => {
347
+ if (!reply.sent) {
348
+ reply.status(500).headers({
349
+ "Cache-Control": "no-store"
350
+ }).send(
351
+ /**
352
+ * When we are sending the error in the response, we cannot send the whole error object, as it might contain some sensitive data.
353
+ */
354
+ JSON.stringify({
355
+ message: error.message,
356
+ code: error.code,
357
+ data: error.data
358
+ }));
359
+ } else {
360
+ console.warn("Reply already sent, cannot send the result (handler:addHook:onError).");
361
+ }
362
+ const handler = middleware(plugins.map(pl => {
393
363
  return (context, error, next) => {
394
364
  return pl.handle(context, error, next);
395
365
  };
@@ -397,20 +367,38 @@ const createHandler = params => {
397
367
  await handler(app.webiny, error);
398
368
  return reply;
399
369
  });
370
+
371
+ /**
372
+ * Apply response headers modifier plugins.
373
+ */
374
+ app.addHook("onSend", async (request, reply, input) => {
375
+ modifyResponseHeaders(app, request, reply);
376
+ const plugins = app.webiny.plugins.byType(OnRequestResponseSendPlugin.type);
377
+ let payload = input;
378
+ for (const plugin of plugins) {
379
+ payload = await plugin.exec(request, reply, payload);
380
+ }
381
+ return payload;
382
+ });
383
+
400
384
  /**
401
385
  * We need to output the benchmark results at the end of the request in both response and timeout cases
402
386
  */
403
387
  app.addHook("onResponse", async () => {
404
388
  await context.benchmark.output();
405
389
  });
406
- app.addHook("onTimeout", async () => {
390
+ app.addHook("onTimeout", async (request, reply) => {
391
+ const plugins = app.webiny.plugins.byType(OnRequestTimeoutPlugin.type);
392
+ for (const plugin of plugins) {
393
+ await plugin.exec(request, reply);
394
+ }
407
395
  await context.benchmark.output();
408
396
  });
409
397
 
410
398
  /**
411
399
  * With these plugins we give users possibility to do anything they want on our fastify instance.
412
400
  */
413
- const modifyPlugins = app.webiny.plugins.byType(_ModifyFastifyPlugin.ModifyFastifyPlugin.type);
401
+ const modifyPlugins = app.webiny.plugins.byType(ModifyFastifyPlugin.type);
414
402
  let modifyFastifyPluginName;
415
403
  try {
416
404
  for (const plugin of modifyPlugins) {
@@ -422,6 +410,34 @@ const createHandler = params => {
422
410
  console.error(stringifyError(ex));
423
411
  throw ex;
424
412
  }
413
+
414
+ /**
415
+ * We have few types of triggers:
416
+ * * Events - EventPlugin
417
+ * * Routes - RoutePlugin
418
+ *
419
+ * Routes are registered in fastify but events must be handled in package which implements cloud specific methods.
420
+ */
421
+ const routePlugins = app.webiny.plugins.byType(RoutePlugin.type);
422
+
423
+ /**
424
+ * Add routes to the system.
425
+ */
426
+ let routePluginName;
427
+ try {
428
+ for (const plugin of routePlugins) {
429
+ routePluginName = plugin.name;
430
+ plugin.cb({
431
+ ...app.webiny.routes,
432
+ context: app.webiny
433
+ });
434
+ }
435
+ } catch (ex) {
436
+ console.error(`Error while running the "RoutePlugin" ${routePluginName ? `(${routePluginName})` : ""} plugin in the beginning of the "createHandler" callable.`);
437
+ console.error(stringifyError(ex));
438
+ throw ex;
439
+ }
425
440
  return app;
426
441
  };
427
- exports.createHandler = createHandler;
442
+
443
+ //# sourceMappingURL=fastify.js.map