@webiny/handler 0.0.0-unstable.615a930a68 → 0.0.0-unstable.61c048f412
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 +18 -9
- package/Context.js +17 -20
- package/Context.js.map +1 -1
- package/PreHandler/IPreHandler.d.ts +9 -0
- package/PreHandler/IPreHandler.js +7 -0
- package/PreHandler/IPreHandler.js.map +1 -0
- package/PreHandler/IfNotOptionsRequest.d.ts +9 -0
- package/PreHandler/IfNotOptionsRequest.js +21 -0
- package/PreHandler/IfNotOptionsRequest.js.map +1 -0
- package/PreHandler/IfOptionsRequest.d.ts +9 -0
- package/PreHandler/IfOptionsRequest.js +21 -0
- package/PreHandler/IfOptionsRequest.js.map +1 -0
- package/PreHandler/PreHandler.d.ts +9 -0
- package/PreHandler/PreHandler.js +17 -0
- package/PreHandler/PreHandler.js.map +1 -0
- package/PreHandler/ProcessBeforeHandlerPlugins.d.ts +10 -0
- package/PreHandler/ProcessBeforeHandlerPlugins.js +24 -0
- package/PreHandler/ProcessBeforeHandlerPlugins.js.map +1 -0
- package/PreHandler/ProcessContextPlugins.d.ts +10 -0
- package/PreHandler/ProcessContextPlugins.js +24 -0
- package/PreHandler/ProcessContextPlugins.js.map +1 -0
- package/PreHandler/ProcessHandlerOnRequestPlugins.d.ts +10 -0
- package/PreHandler/ProcessHandlerOnRequestPlugins.js +26 -0
- package/PreHandler/ProcessHandlerOnRequestPlugins.js.map +1 -0
- package/PreHandler/SendEarlyOptionsResponse.d.ts +9 -0
- package/PreHandler/SendEarlyOptionsResponse.js +31 -0
- package/PreHandler/SendEarlyOptionsResponse.js.map +1 -0
- package/PreHandler/SetDefaultHeaders.d.ts +9 -0
- package/PreHandler/SetDefaultHeaders.js +57 -0
- package/PreHandler/SetDefaultHeaders.js.map +1 -0
- package/README.md +10 -14
- package/ResponseHeaders.d.ts +24 -0
- package/ResponseHeaders.js +39 -0
- package/ResponseHeaders.js.map +1 -0
- package/abstractions/Reply.d.ts +5 -0
- package/abstractions/Reply.js +4 -0
- package/abstractions/Reply.js.map +1 -0
- package/abstractions/Request.d.ts +5 -0
- package/abstractions/Request.js +4 -0
- package/abstractions/Request.js.map +1 -0
- package/fastify.d.ts +8 -5
- package/fastify.js +286 -186
- package/fastify.js.map +1 -1
- package/index.d.ts +17 -8
- package/index.js +20 -109
- package/index.js.map +1 -1
- package/package.json +16 -26
- package/plugins/BeforeHandlerPlugin.d.ts +1 -1
- package/plugins/BeforeHandlerPlugin.js +5 -24
- package/plugins/BeforeHandlerPlugin.js.map +1 -1
- package/plugins/EventPlugin.d.ts +3 -3
- package/plugins/EventPlugin.js +5 -22
- package/plugins/EventPlugin.js.map +1 -1
- package/plugins/HandlerErrorPlugin.d.ts +6 -3
- package/plugins/HandlerErrorPlugin.js +5 -23
- package/plugins/HandlerErrorPlugin.js.map +1 -1
- package/plugins/HandlerOnRequestPlugin.d.ts +21 -0
- package/plugins/HandlerOnRequestPlugin.js +24 -0
- package/plugins/HandlerOnRequestPlugin.js.map +1 -0
- package/plugins/HandlerResultPlugin.d.ts +1 -1
- package/plugins/HandlerResultPlugin.js +5 -23
- package/plugins/HandlerResultPlugin.js.map +1 -1
- package/plugins/ModifyFastifyPlugin.d.ts +13 -0
- package/plugins/ModifyFastifyPlugin.js +16 -0
- package/plugins/ModifyFastifyPlugin.js.map +1 -0
- package/plugins/ModifyResponseHeadersPlugin.d.ts +14 -0
- package/plugins/ModifyResponseHeadersPlugin.js +16 -0
- package/plugins/ModifyResponseHeadersPlugin.js.map +1 -0
- package/plugins/OnRequestResponseSendPlugin.d.ts +26 -0
- package/plugins/OnRequestResponseSendPlugin.js +31 -0
- package/plugins/OnRequestResponseSendPlugin.js.map +1 -0
- package/plugins/OnRequestTimeoutPlugin.d.ts +12 -0
- package/plugins/OnRequestTimeoutPlugin.js +16 -0
- package/plugins/OnRequestTimeoutPlugin.js.map +1 -0
- package/plugins/RoutePlugin.d.ts +2 -2
- package/plugins/RoutePlugin.js +5 -22
- package/plugins/RoutePlugin.js.map +1 -1
- package/stringifyError.d.ts +5 -0
- package/stringifyError.js +20 -0
- package/stringifyError.js.map +1 -0
- package/suppressPunycodeWarnings.d.ts +1 -0
- package/suppressPunycodeWarnings.js +10 -0
- package/suppressPunycodeWarnings.js.map +1 -0
- package/types.d.ts +14 -13
- package/types.js +2 -18
- package/types.js.map +1 -1
- package/middleware.d.ts +0 -4
- package/middleware.js +0 -51
- package/middleware.js.map +0 -1
|
@@ -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
|
-
|
|
2
|
-
import {
|
|
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) =>
|
|
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,80 +1,48 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
}
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
const DEFAULT_HEADERS = (0, _objectSpread2.default)({
|
|
39
|
-
"Cache-Control": "no-store",
|
|
40
|
-
"Content-Type": "application/json; charset=utf-8",
|
|
41
|
-
"Access-Control-Allow-Origin": "*",
|
|
42
|
-
"Access-Control-Allow-Headers": "*",
|
|
43
|
-
"Access-Control-Allow-Methods": "OPTIONS,POST,GET,DELETE,PUT,PATCH"
|
|
44
|
-
}, (0, _utils.getWebinyVersionHeaders)());
|
|
45
|
-
|
|
46
|
-
const getDefaultHeaders = routes => {
|
|
47
|
-
/**
|
|
48
|
-
* If we are accepting all headers, just output that one.
|
|
49
|
-
*/
|
|
50
|
-
const keys = Object.keys(routes);
|
|
51
|
-
const all = keys.every(key => routes[key].length > 0);
|
|
52
|
-
|
|
53
|
-
if (all) {
|
|
54
|
-
return (0, _objectSpread2.default)((0, _objectSpread2.default)({}, DEFAULT_HEADERS), {}, {
|
|
55
|
-
"Access-Control-Allow-Methods": "*"
|
|
56
|
-
});
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
return (0, _objectSpread2.default)((0, _objectSpread2.default)({}, DEFAULT_HEADERS), {}, {
|
|
60
|
-
"Access-Control-Allow-Methods": keys.filter(key => {
|
|
61
|
-
const type = key;
|
|
62
|
-
|
|
63
|
-
if (!routes[type] || Array.isArray(routes[type]) === false) {
|
|
64
|
-
return false;
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
return routes[type].length > 0;
|
|
68
|
-
}).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);
|
|
69
37
|
});
|
|
70
|
-
};
|
|
71
38
|
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
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);
|
|
75
44
|
};
|
|
76
|
-
|
|
77
|
-
const createHandler = params => {
|
|
45
|
+
export const createHandler = params => {
|
|
78
46
|
const definedRoutes = {
|
|
79
47
|
POST: [],
|
|
80
48
|
GET: [],
|
|
@@ -91,92 +59,89 @@ const createHandler = params => {
|
|
|
91
59
|
PROPPATCH: [],
|
|
92
60
|
SEARCH: [],
|
|
93
61
|
TRACE: [],
|
|
94
|
-
UNLOCK: []
|
|
62
|
+
UNLOCK: [],
|
|
63
|
+
REPORT: [],
|
|
64
|
+
MKCALENDAR: []
|
|
95
65
|
};
|
|
96
|
-
|
|
97
66
|
const throwOnDefinedRoute = (type, path, options) => {
|
|
98
67
|
if (type === "ALL") {
|
|
99
|
-
const all = Object.keys(definedRoutes).
|
|
68
|
+
const all = Object.keys(definedRoutes).find(k => {
|
|
69
|
+
const key = k.toUpperCase();
|
|
100
70
|
const routes = definedRoutes[key];
|
|
101
71
|
return routes.includes(path);
|
|
102
72
|
});
|
|
103
|
-
|
|
104
73
|
if (!all) {
|
|
105
74
|
return;
|
|
106
75
|
}
|
|
107
|
-
|
|
108
|
-
|
|
76
|
+
console.error(`Error while registering onAll route. One of the routes is already defined.`);
|
|
77
|
+
console.error(JSON.stringify(all));
|
|
78
|
+
throw new WebinyError(`You cannot override a route with onAll() method, please remove unnecessary route from the system.`, "OVERRIDE_ROUTE_ERROR", {
|
|
109
79
|
type,
|
|
110
80
|
path
|
|
111
81
|
});
|
|
112
82
|
} else if (definedRoutes[type].includes(path) === false) {
|
|
113
83
|
return;
|
|
114
|
-
} else if (
|
|
84
|
+
} else if (options?.override === true) {
|
|
115
85
|
return;
|
|
116
86
|
}
|
|
117
|
-
|
|
118
|
-
throw new
|
|
87
|
+
console.error(`Error while trying to override route: [${type}] ${path}`);
|
|
88
|
+
throw new WebinyError(`When you are trying to override existing route, you must send "override" parameter when adding that route.`, "OVERRIDE_ROUTE_ERROR", {
|
|
119
89
|
type,
|
|
120
90
|
path
|
|
121
91
|
});
|
|
122
92
|
};
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
const type = inputType.toUpperCase();
|
|
126
|
-
|
|
93
|
+
const addDefinedRoute = (input, path) => {
|
|
94
|
+
const type = input.toUpperCase();
|
|
127
95
|
if (!definedRoutes[type]) {
|
|
128
96
|
return;
|
|
129
97
|
} else if (definedRoutes[type].includes(path)) {
|
|
130
98
|
return;
|
|
131
99
|
}
|
|
132
|
-
|
|
133
100
|
definedRoutes[type].push(path);
|
|
134
101
|
};
|
|
102
|
+
|
|
135
103
|
/**
|
|
136
104
|
* We must attach the server to our internal context if we want to have it accessible.
|
|
137
105
|
*/
|
|
106
|
+
const app = fastify({
|
|
107
|
+
bodyLimit: 536870912,
|
|
108
|
+
// 512MB
|
|
109
|
+
disableRequestLogging: true,
|
|
110
|
+
allowErrorHandlerOverride: true,
|
|
111
|
+
...(params.options || {})
|
|
112
|
+
});
|
|
138
113
|
|
|
139
|
-
|
|
140
|
-
const app = (0, _fastify.default)((0, _objectSpread2.default)({}, params.options || {}));
|
|
141
114
|
/**
|
|
142
|
-
* We need to register routes in our system
|
|
115
|
+
* We need to register routes in our system to output headers later on, and disallow route overriding.
|
|
143
116
|
*/
|
|
144
|
-
|
|
145
117
|
app.addHook("onRoute", route => {
|
|
146
118
|
const method = route.method;
|
|
147
|
-
|
|
148
119
|
if (Array.isArray(method)) {
|
|
149
120
|
for (const m of method) {
|
|
150
121
|
addDefinedRoute(m, route.path);
|
|
151
122
|
}
|
|
152
|
-
|
|
153
123
|
return;
|
|
154
124
|
}
|
|
155
|
-
|
|
156
125
|
addDefinedRoute(method, route.path);
|
|
157
126
|
});
|
|
158
127
|
/**
|
|
159
128
|
* ############################
|
|
160
129
|
* Register the Fastify plugins.
|
|
161
130
|
*/
|
|
162
|
-
|
|
163
131
|
/**
|
|
164
132
|
* Package @fastify/cookie
|
|
165
133
|
*
|
|
166
134
|
* https://github.com/fastify/fastify-cookie
|
|
167
135
|
*/
|
|
168
|
-
|
|
169
|
-
app.register(_cookie.default, {
|
|
136
|
+
app.register(fastifyCookie, {
|
|
170
137
|
parseOptions: {} // options for parsing cookies
|
|
171
|
-
|
|
172
138
|
});
|
|
173
139
|
/**
|
|
174
140
|
* Package @fastify/compress
|
|
175
141
|
*
|
|
176
142
|
* https://github.com/fastify/fastify-compress
|
|
177
143
|
*/
|
|
178
|
-
|
|
179
|
-
app.register(_compress.default, {
|
|
144
|
+
app.register(fastifyCompress, {
|
|
180
145
|
global: true,
|
|
181
146
|
threshold: 1024,
|
|
182
147
|
onUnsupportedEncoding: (encoding, _, reply) => {
|
|
@@ -188,7 +153,6 @@ const createHandler = params => {
|
|
|
188
153
|
/**
|
|
189
154
|
* Route helpers - mostly for users.
|
|
190
155
|
*/
|
|
191
|
-
|
|
192
156
|
const routes = {
|
|
193
157
|
defined: definedRoutes,
|
|
194
158
|
onPost: (path, handler, options) => {
|
|
@@ -224,120 +188,256 @@ const createHandler = params => {
|
|
|
224
188
|
app.head(path, handler);
|
|
225
189
|
}
|
|
226
190
|
};
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
/**
|
|
230
|
-
* We must have handlerClient by default.
|
|
231
|
-
* And it must be one of the first context plugins applied.
|
|
232
|
-
*/
|
|
233
|
-
(0, _handlerClient.createHandlerClient)(), ...(params.plugins || [])],
|
|
234
|
-
|
|
235
|
-
/**
|
|
236
|
-
* Inserted via webpack on build time.
|
|
237
|
-
*/
|
|
238
|
-
WEBINY_VERSION: process.env.WEBINY_VERSION,
|
|
239
|
-
server: app,
|
|
240
|
-
routes
|
|
241
|
-
});
|
|
191
|
+
let context;
|
|
192
|
+
const plugins = new PluginsContainer([
|
|
242
193
|
/**
|
|
243
|
-
* We
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
app.decorate("webiny", context);
|
|
247
|
-
/**
|
|
248
|
-
* We have few types of triggers:
|
|
249
|
-
* * Events - EventPlugin
|
|
250
|
-
* * Routes - RoutePlugin
|
|
251
|
-
*
|
|
252
|
-
* Routes are registered in fastify but events must be handled in package which implements cloud specific methods.
|
|
194
|
+
* We must have handlerClient by default.
|
|
195
|
+
* And it must be one of the first context plugins applied.
|
|
253
196
|
*/
|
|
197
|
+
createHandlerClient()]);
|
|
198
|
+
plugins.merge(params.plugins || []);
|
|
199
|
+
try {
|
|
200
|
+
context = new Context({
|
|
201
|
+
plugins,
|
|
202
|
+
/**
|
|
203
|
+
* Inserted via webpack at build time.
|
|
204
|
+
*/
|
|
205
|
+
WEBINY_VERSION: process.env.WEBINY_VERSION,
|
|
206
|
+
routes
|
|
207
|
+
});
|
|
208
|
+
} catch (ex) {
|
|
209
|
+
console.error(`Error while constructing the Context.`);
|
|
210
|
+
console.error(stringifyError(ex));
|
|
211
|
+
throw ex;
|
|
212
|
+
}
|
|
254
213
|
|
|
255
|
-
const routePlugins = app.webiny.plugins.byType(_RoutePlugin.RoutePlugin.type);
|
|
256
214
|
/**
|
|
257
|
-
*
|
|
215
|
+
* We are attaching our custom context to webiny variable on the fastify app, so it is accessible everywhere.
|
|
258
216
|
*/
|
|
217
|
+
app.decorate("webiny", context);
|
|
259
218
|
|
|
260
|
-
for (const plugin of routePlugins) {
|
|
261
|
-
plugin.cb((0, _objectSpread2.default)((0, _objectSpread2.default)({}, app.webiny.routes), {}, {
|
|
262
|
-
context: app.webiny
|
|
263
|
-
}));
|
|
264
|
-
}
|
|
265
219
|
/**
|
|
266
|
-
*
|
|
267
|
-
*
|
|
220
|
+
* To prevent Unsupported Media Type errors on OPTIONS requests with a body,
|
|
221
|
+
* we need to have a custom parser
|
|
268
222
|
*/
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
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);
|
|
276
229
|
return;
|
|
277
230
|
}
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
raw.setHeader(key, headers[key]);
|
|
231
|
+
try {
|
|
232
|
+
const json = typeof body === "string" ? body : body.toString("utf8");
|
|
233
|
+
done(null, JSON.parse(json));
|
|
234
|
+
} catch (err) {
|
|
235
|
+
done(err);
|
|
284
236
|
}
|
|
285
|
-
|
|
286
|
-
raw.end("");
|
|
287
237
|
});
|
|
288
|
-
app.addHook("preParsing", async request => {
|
|
289
|
-
app.webiny.request = request;
|
|
290
|
-
const plugins = app.webiny.plugins.byType(_api.ContextPlugin.type);
|
|
291
238
|
|
|
292
|
-
for (const plugin of plugins) {
|
|
293
|
-
await plugin.apply(app.webiny);
|
|
294
|
-
}
|
|
295
|
-
});
|
|
296
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.
|
|
297
242
|
*
|
|
243
|
+
* @see https://fastify.dev/docs/latest/Reference/ContentTypeParser/#content-type-parser
|
|
298
244
|
*/
|
|
299
|
-
|
|
300
|
-
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
for (const plugin of plugins) {
|
|
304
|
-
await plugin.apply(app.webiny);
|
|
245
|
+
app.addHook("onRequest", async request => {
|
|
246
|
+
if (request.method === "OPTIONS" && request.body === undefined) {
|
|
247
|
+
request.headers["content-type"] = undefined;
|
|
305
248
|
}
|
|
306
249
|
});
|
|
250
|
+
|
|
307
251
|
/**
|
|
308
|
-
*
|
|
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`
|
|
309
258
|
*/
|
|
259
|
+
app.addHook("preHandler", async (request, reply) => {
|
|
260
|
+
app.webiny.request = request;
|
|
261
|
+
app.webiny.reply = reply;
|
|
310
262
|
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
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);
|
|
282
|
+
let name;
|
|
283
|
+
try {
|
|
284
|
+
for (const plugin of plugins) {
|
|
285
|
+
name = plugin.name;
|
|
286
|
+
await plugin.handle(app.webiny, payload);
|
|
287
|
+
}
|
|
288
|
+
} catch (ex) {
|
|
289
|
+
console.error(`Error while running the "HandlerResultPlugin" ${name ? `(${name})` : ""} plugin in the preSerialization hook.`);
|
|
290
|
+
console.error(stringifyError(ex));
|
|
291
|
+
throw ex;
|
|
316
292
|
}
|
|
317
|
-
|
|
318
293
|
return payload;
|
|
319
|
-
};
|
|
320
|
-
|
|
321
|
-
|
|
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
|
+
}
|
|
319
|
+
return reply.status(500).headers({
|
|
320
|
+
"Cache-Control": "no-store"
|
|
321
|
+
}).send(
|
|
322
|
+
/**
|
|
323
|
+
* When we are sending the error in the response, we cannot send the whole error object, as it might contain some sensitive data.
|
|
324
|
+
*/
|
|
325
|
+
JSON.stringify({
|
|
326
|
+
message: error.message,
|
|
327
|
+
code: error.code,
|
|
328
|
+
data: error.data
|
|
329
|
+
}));
|
|
330
|
+
});
|
|
322
331
|
app.addHook("onError", async (_, reply, error) => {
|
|
323
|
-
const plugins = app.webiny.plugins.byType(
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
332
|
+
const plugins = app.webiny.plugins.byType(HandlerErrorPlugin.type);
|
|
333
|
+
/**
|
|
334
|
+
* Log error to cloud, as these can be extremely annoying to debug!
|
|
335
|
+
*/
|
|
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
|
+
}
|
|
344
|
+
/**
|
|
345
|
+
* IMPORTANT! Do not send anything if reply was already sent.
|
|
346
|
+
*/
|
|
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 => {
|
|
331
363
|
return (context, error, next) => {
|
|
332
364
|
return pl.handle(context, error, next);
|
|
333
365
|
};
|
|
334
366
|
}));
|
|
335
367
|
await handler(app.webiny, error);
|
|
336
|
-
return reply
|
|
337
|
-
|
|
338
|
-
|
|
368
|
+
return reply;
|
|
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;
|
|
339
382
|
});
|
|
383
|
+
|
|
384
|
+
/**
|
|
385
|
+
* We need to output the benchmark results at the end of the request in both response and timeout cases
|
|
386
|
+
*/
|
|
387
|
+
app.addHook("onResponse", async () => {
|
|
388
|
+
await context.benchmark.output();
|
|
389
|
+
});
|
|
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
|
+
}
|
|
395
|
+
await context.benchmark.output();
|
|
396
|
+
});
|
|
397
|
+
|
|
398
|
+
/**
|
|
399
|
+
* With these plugins we give users possibility to do anything they want on our fastify instance.
|
|
400
|
+
*/
|
|
401
|
+
const modifyPlugins = app.webiny.plugins.byType(ModifyFastifyPlugin.type);
|
|
402
|
+
let modifyFastifyPluginName;
|
|
403
|
+
try {
|
|
404
|
+
for (const plugin of modifyPlugins) {
|
|
405
|
+
modifyFastifyPluginName = plugin.name;
|
|
406
|
+
plugin.modify(app);
|
|
407
|
+
}
|
|
408
|
+
} catch (ex) {
|
|
409
|
+
console.error(`Error while running the "ModifyFastifyPlugin" ${modifyFastifyPluginName ? `(${modifyFastifyPluginName})` : ""} plugin in the end of the "createHandler" callable.`);
|
|
410
|
+
console.error(stringifyError(ex));
|
|
411
|
+
throw ex;
|
|
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
|
+
}
|
|
340
440
|
return app;
|
|
341
441
|
};
|
|
342
442
|
|
|
343
|
-
|
|
443
|
+
//# sourceMappingURL=fastify.js.map
|