@whatwg-node/server 0.7.7 → 0.8.0
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/package.json +5 -2
- package/createServerAdapter.d.ts +0 -9
- package/index.d.ts +0 -7
- package/index.js +0 -666
- package/index.mjs +0 -643
- package/plugins/types.d.ts +0 -21
- package/plugins/useCors.d.ts +0 -13
- package/plugins/useErrorHandling.d.ts +0 -11
- package/types.d.ts +0 -56
- package/utils.d.ts +0 -33
package/index.js
DELETED
|
@@ -1,666 +0,0 @@
|
|
|
1
|
-
'use strict';
|
|
2
|
-
|
|
3
|
-
Object.defineProperty(exports, '__esModule', { value: true });
|
|
4
|
-
|
|
5
|
-
const DefaultFetchAPI = require('@whatwg-node/fetch');
|
|
6
|
-
|
|
7
|
-
function isAsyncIterable(body) {
|
|
8
|
-
return (body != null && typeof body === 'object' && typeof body[Symbol.asyncIterator] === 'function');
|
|
9
|
-
}
|
|
10
|
-
function getPort(nodeRequest) {
|
|
11
|
-
var _a, _b, _c, _d, _e;
|
|
12
|
-
if ((_a = nodeRequest.socket) === null || _a === void 0 ? void 0 : _a.localPort) {
|
|
13
|
-
return (_b = nodeRequest.socket) === null || _b === void 0 ? void 0 : _b.localPort;
|
|
14
|
-
}
|
|
15
|
-
const hostInHeader = ((_c = nodeRequest.headers) === null || _c === void 0 ? void 0 : _c[':authority']) || ((_d = nodeRequest.headers) === null || _d === void 0 ? void 0 : _d.host);
|
|
16
|
-
const portInHeader = (_e = hostInHeader === null || hostInHeader === void 0 ? void 0 : hostInHeader.split(':')) === null || _e === void 0 ? void 0 : _e[1];
|
|
17
|
-
if (portInHeader) {
|
|
18
|
-
return portInHeader;
|
|
19
|
-
}
|
|
20
|
-
return 80;
|
|
21
|
-
}
|
|
22
|
-
function getHostnameWithPort(nodeRequest) {
|
|
23
|
-
var _a, _b, _c, _d, _e;
|
|
24
|
-
if ((_a = nodeRequest.headers) === null || _a === void 0 ? void 0 : _a[':authority']) {
|
|
25
|
-
return (_b = nodeRequest.headers) === null || _b === void 0 ? void 0 : _b[':authority'];
|
|
26
|
-
}
|
|
27
|
-
if ((_c = nodeRequest.headers) === null || _c === void 0 ? void 0 : _c.host) {
|
|
28
|
-
return (_d = nodeRequest.headers) === null || _d === void 0 ? void 0 : _d.host;
|
|
29
|
-
}
|
|
30
|
-
const port = getPort(nodeRequest);
|
|
31
|
-
if (nodeRequest.hostname) {
|
|
32
|
-
return nodeRequest.hostname + ':' + port;
|
|
33
|
-
}
|
|
34
|
-
const localIp = (_e = nodeRequest.socket) === null || _e === void 0 ? void 0 : _e.localAddress;
|
|
35
|
-
if (localIp && !(localIp === null || localIp === void 0 ? void 0 : localIp.includes('::')) && !(localIp === null || localIp === void 0 ? void 0 : localIp.includes('ffff'))) {
|
|
36
|
-
return `${localIp}:${port}`;
|
|
37
|
-
}
|
|
38
|
-
return 'localhost';
|
|
39
|
-
}
|
|
40
|
-
function buildFullUrl(nodeRequest) {
|
|
41
|
-
const hostnameWithPort = getHostnameWithPort(nodeRequest);
|
|
42
|
-
const protocol = nodeRequest.protocol || 'http';
|
|
43
|
-
const endpoint = nodeRequest.originalUrl || nodeRequest.url || '/graphql';
|
|
44
|
-
return `${protocol}://${hostnameWithPort}${endpoint}`;
|
|
45
|
-
}
|
|
46
|
-
function isRequestBody(body) {
|
|
47
|
-
const stringTag = body[Symbol.toStringTag];
|
|
48
|
-
if (typeof body === 'string' ||
|
|
49
|
-
stringTag === 'Uint8Array' ||
|
|
50
|
-
stringTag === 'Blob' ||
|
|
51
|
-
stringTag === 'FormData' ||
|
|
52
|
-
stringTag === 'URLSearchParams' ||
|
|
53
|
-
isAsyncIterable(body)) {
|
|
54
|
-
return true;
|
|
55
|
-
}
|
|
56
|
-
return false;
|
|
57
|
-
}
|
|
58
|
-
function normalizeNodeRequest(nodeRequest, RequestCtor) {
|
|
59
|
-
var _a;
|
|
60
|
-
const rawRequest = nodeRequest.raw || nodeRequest.req || nodeRequest;
|
|
61
|
-
let fullUrl = buildFullUrl(rawRequest);
|
|
62
|
-
if (nodeRequest.query) {
|
|
63
|
-
const url = new DefaultFetchAPI.URL(fullUrl);
|
|
64
|
-
for (const key in nodeRequest.query) {
|
|
65
|
-
url.searchParams.set(key, nodeRequest.query[key]);
|
|
66
|
-
}
|
|
67
|
-
fullUrl = url.toString();
|
|
68
|
-
}
|
|
69
|
-
if (nodeRequest.method === 'GET' || nodeRequest.method === 'HEAD') {
|
|
70
|
-
return new RequestCtor(fullUrl, {
|
|
71
|
-
method: nodeRequest.method,
|
|
72
|
-
headers: nodeRequest.headers,
|
|
73
|
-
});
|
|
74
|
-
}
|
|
75
|
-
/**
|
|
76
|
-
* Some Node server frameworks like Serverless Express sends a dummy object with body but as a Buffer not string
|
|
77
|
-
* so we do those checks to see is there something we can use directly as BodyInit
|
|
78
|
-
* because the presence of body means the request stream is already consumed and,
|
|
79
|
-
* rawRequest cannot be used as BodyInit/ReadableStream by Fetch API in this case.
|
|
80
|
-
*/
|
|
81
|
-
const maybeParsedBody = nodeRequest.body;
|
|
82
|
-
if (maybeParsedBody != null && Object.keys(maybeParsedBody).length > 0) {
|
|
83
|
-
if (isRequestBody(maybeParsedBody)) {
|
|
84
|
-
return new RequestCtor(fullUrl, {
|
|
85
|
-
method: nodeRequest.method,
|
|
86
|
-
headers: nodeRequest.headers,
|
|
87
|
-
body: maybeParsedBody,
|
|
88
|
-
});
|
|
89
|
-
}
|
|
90
|
-
const request = new RequestCtor(fullUrl, {
|
|
91
|
-
method: nodeRequest.method,
|
|
92
|
-
headers: nodeRequest.headers,
|
|
93
|
-
});
|
|
94
|
-
if (!((_a = request.headers.get('content-type')) === null || _a === void 0 ? void 0 : _a.includes('json'))) {
|
|
95
|
-
request.headers.set('content-type', 'application/json');
|
|
96
|
-
}
|
|
97
|
-
return new Proxy(request, {
|
|
98
|
-
get: (target, prop, receiver) => {
|
|
99
|
-
switch (prop) {
|
|
100
|
-
case 'json':
|
|
101
|
-
return async () => maybeParsedBody;
|
|
102
|
-
case 'text':
|
|
103
|
-
return async () => JSON.stringify(maybeParsedBody);
|
|
104
|
-
default:
|
|
105
|
-
return Reflect.get(target, prop, receiver);
|
|
106
|
-
}
|
|
107
|
-
},
|
|
108
|
-
});
|
|
109
|
-
}
|
|
110
|
-
// perf: instead of spreading the object, we can just pass it as is and it performs better
|
|
111
|
-
return new RequestCtor(fullUrl, {
|
|
112
|
-
method: nodeRequest.method,
|
|
113
|
-
headers: nodeRequest.headers,
|
|
114
|
-
body: rawRequest,
|
|
115
|
-
});
|
|
116
|
-
}
|
|
117
|
-
function isReadable(stream) {
|
|
118
|
-
return stream.read != null;
|
|
119
|
-
}
|
|
120
|
-
function isNodeRequest(request) {
|
|
121
|
-
return isReadable(request);
|
|
122
|
-
}
|
|
123
|
-
function isServerResponse(stream) {
|
|
124
|
-
// Check all used functions are defined
|
|
125
|
-
return (stream != null &&
|
|
126
|
-
stream.setHeader != null &&
|
|
127
|
-
stream.end != null &&
|
|
128
|
-
stream.once != null &&
|
|
129
|
-
stream.write != null);
|
|
130
|
-
}
|
|
131
|
-
function isReadableStream(stream) {
|
|
132
|
-
return stream != null && stream.getReader != null;
|
|
133
|
-
}
|
|
134
|
-
function isFetchEvent(event) {
|
|
135
|
-
return event != null && event.request != null && event.respondWith != null;
|
|
136
|
-
}
|
|
137
|
-
function configureSocket(rawRequest) {
|
|
138
|
-
var _a, _b, _c, _d, _e, _f;
|
|
139
|
-
(_b = (_a = rawRequest === null || rawRequest === void 0 ? void 0 : rawRequest.socket) === null || _a === void 0 ? void 0 : _a.setTimeout) === null || _b === void 0 ? void 0 : _b.call(_a, 0);
|
|
140
|
-
(_d = (_c = rawRequest === null || rawRequest === void 0 ? void 0 : rawRequest.socket) === null || _c === void 0 ? void 0 : _c.setNoDelay) === null || _d === void 0 ? void 0 : _d.call(_c, true);
|
|
141
|
-
(_f = (_e = rawRequest === null || rawRequest === void 0 ? void 0 : rawRequest.socket) === null || _e === void 0 ? void 0 : _e.setKeepAlive) === null || _f === void 0 ? void 0 : _f.call(_e, true);
|
|
142
|
-
}
|
|
143
|
-
function endResponse(serverResponse) {
|
|
144
|
-
// @ts-expect-error Avoid arguments adaptor trampoline https://v8.dev/blog/adaptor-frame
|
|
145
|
-
serverResponse.end(null, null, null);
|
|
146
|
-
}
|
|
147
|
-
function getHeadersObj(headers) {
|
|
148
|
-
return new Proxy({}, {
|
|
149
|
-
get(_target, prop) {
|
|
150
|
-
return headers.get(prop) || undefined;
|
|
151
|
-
},
|
|
152
|
-
set(_target, prop, value) {
|
|
153
|
-
headers.set(prop, value);
|
|
154
|
-
return true;
|
|
155
|
-
},
|
|
156
|
-
has(_target, prop) {
|
|
157
|
-
return headers.has(prop);
|
|
158
|
-
},
|
|
159
|
-
deleteProperty(_target, prop) {
|
|
160
|
-
headers.delete(prop);
|
|
161
|
-
return true;
|
|
162
|
-
},
|
|
163
|
-
ownKeys() {
|
|
164
|
-
const keys = [];
|
|
165
|
-
headers.forEach((_, key) => keys.push(key));
|
|
166
|
-
return keys;
|
|
167
|
-
},
|
|
168
|
-
getOwnPropertyDescriptor() {
|
|
169
|
-
return {
|
|
170
|
-
enumerable: true,
|
|
171
|
-
configurable: true,
|
|
172
|
-
};
|
|
173
|
-
},
|
|
174
|
-
});
|
|
175
|
-
}
|
|
176
|
-
async function sendNodeResponse(fetchResponse, serverResponse, nodeRequest) {
|
|
177
|
-
const headersObj = getHeadersObj(fetchResponse.headers);
|
|
178
|
-
serverResponse.writeHead(fetchResponse.status, fetchResponse.statusText, headersObj);
|
|
179
|
-
// eslint-disable-next-line no-async-promise-executor
|
|
180
|
-
return new Promise(async (resolve) => {
|
|
181
|
-
serverResponse.once('close', resolve);
|
|
182
|
-
// Our Node-fetch enhancements
|
|
183
|
-
if ('bodyType' in fetchResponse &&
|
|
184
|
-
fetchResponse.bodyType != null &&
|
|
185
|
-
(fetchResponse.bodyType === 'String' || fetchResponse.bodyType === 'Uint8Array')) {
|
|
186
|
-
// @ts-expect-error http and http2 writes are actually compatible
|
|
187
|
-
serverResponse.write(fetchResponse.bodyInit);
|
|
188
|
-
endResponse(serverResponse);
|
|
189
|
-
return;
|
|
190
|
-
}
|
|
191
|
-
// Other fetch implementations
|
|
192
|
-
const fetchBody = fetchResponse.body;
|
|
193
|
-
if (fetchBody == null) {
|
|
194
|
-
endResponse(serverResponse);
|
|
195
|
-
return;
|
|
196
|
-
}
|
|
197
|
-
if (fetchBody[Symbol.toStringTag] === 'Uint8Array') {
|
|
198
|
-
serverResponse
|
|
199
|
-
// @ts-expect-error http and http2 writes are actually compatible
|
|
200
|
-
.write(fetchBody);
|
|
201
|
-
endResponse(serverResponse);
|
|
202
|
-
return;
|
|
203
|
-
}
|
|
204
|
-
configureSocket(nodeRequest);
|
|
205
|
-
if (isReadable(fetchBody)) {
|
|
206
|
-
serverResponse.once('close', () => {
|
|
207
|
-
fetchBody.destroy();
|
|
208
|
-
});
|
|
209
|
-
fetchBody.pipe(serverResponse);
|
|
210
|
-
return;
|
|
211
|
-
}
|
|
212
|
-
if (isAsyncIterable(fetchBody)) {
|
|
213
|
-
for await (const chunk of fetchBody) {
|
|
214
|
-
if (!serverResponse
|
|
215
|
-
// @ts-expect-error http and http2 writes are actually compatible
|
|
216
|
-
.write(chunk)) {
|
|
217
|
-
break;
|
|
218
|
-
}
|
|
219
|
-
}
|
|
220
|
-
endResponse(serverResponse);
|
|
221
|
-
}
|
|
222
|
-
});
|
|
223
|
-
}
|
|
224
|
-
function isRequestInit(val) {
|
|
225
|
-
return (val != null &&
|
|
226
|
-
typeof val === 'object' &&
|
|
227
|
-
('body' in val ||
|
|
228
|
-
'cache' in val ||
|
|
229
|
-
'credentials' in val ||
|
|
230
|
-
'headers' in val ||
|
|
231
|
-
'integrity' in val ||
|
|
232
|
-
'keepalive' in val ||
|
|
233
|
-
'method' in val ||
|
|
234
|
-
'mode' in val ||
|
|
235
|
-
'redirect' in val ||
|
|
236
|
-
'referrer' in val ||
|
|
237
|
-
'referrerPolicy' in val ||
|
|
238
|
-
'signal' in val ||
|
|
239
|
-
'window' in val));
|
|
240
|
-
}
|
|
241
|
-
|
|
242
|
-
/* eslint-disable @typescript-eslint/ban-types */
|
|
243
|
-
async function handleWaitUntils(waitUntilPromises) {
|
|
244
|
-
const waitUntils = await Promise.allSettled(waitUntilPromises);
|
|
245
|
-
waitUntils.forEach(waitUntil => {
|
|
246
|
-
if (waitUntil.status === 'rejected') {
|
|
247
|
-
console.error(waitUntil.reason);
|
|
248
|
-
}
|
|
249
|
-
});
|
|
250
|
-
}
|
|
251
|
-
// Required for envs like nextjs edge runtime
|
|
252
|
-
function isRequestAccessible(serverContext) {
|
|
253
|
-
try {
|
|
254
|
-
return !!(serverContext === null || serverContext === void 0 ? void 0 : serverContext.request);
|
|
255
|
-
}
|
|
256
|
-
catch (_a) {
|
|
257
|
-
return false;
|
|
258
|
-
}
|
|
259
|
-
}
|
|
260
|
-
function createServerAdapter(serverAdapterBaseObject, options) {
|
|
261
|
-
const fetchAPI = {
|
|
262
|
-
...DefaultFetchAPI,
|
|
263
|
-
...options === null || options === void 0 ? void 0 : options.fetchAPI,
|
|
264
|
-
};
|
|
265
|
-
const givenHandleRequest = typeof serverAdapterBaseObject === 'function'
|
|
266
|
-
? serverAdapterBaseObject
|
|
267
|
-
: serverAdapterBaseObject.handle;
|
|
268
|
-
const onRequestHooks = [];
|
|
269
|
-
const onResponseHooks = [];
|
|
270
|
-
if ((options === null || options === void 0 ? void 0 : options.plugins) != null) {
|
|
271
|
-
for (const plugin of options.plugins) {
|
|
272
|
-
if (plugin.onRequest) {
|
|
273
|
-
onRequestHooks.push(plugin.onRequest);
|
|
274
|
-
}
|
|
275
|
-
if (plugin.onResponse) {
|
|
276
|
-
onResponseHooks.push(plugin.onResponse);
|
|
277
|
-
}
|
|
278
|
-
}
|
|
279
|
-
}
|
|
280
|
-
async function handleRequest(request, serverContext) {
|
|
281
|
-
let url = new Proxy({}, {
|
|
282
|
-
get: (_target, prop, _receiver) => {
|
|
283
|
-
url = new fetchAPI.URL(request.url, 'http://localhost');
|
|
284
|
-
return Reflect.get(url, prop, url);
|
|
285
|
-
},
|
|
286
|
-
});
|
|
287
|
-
let requestHandler = givenHandleRequest;
|
|
288
|
-
let response;
|
|
289
|
-
for (const onRequestHook of onRequestHooks) {
|
|
290
|
-
await onRequestHook({
|
|
291
|
-
request,
|
|
292
|
-
serverContext,
|
|
293
|
-
fetchAPI,
|
|
294
|
-
url,
|
|
295
|
-
requestHandler,
|
|
296
|
-
setRequestHandler(newRequestHandler) {
|
|
297
|
-
requestHandler = newRequestHandler;
|
|
298
|
-
},
|
|
299
|
-
endResponse(newResponse) {
|
|
300
|
-
response = newResponse;
|
|
301
|
-
},
|
|
302
|
-
});
|
|
303
|
-
if (response) {
|
|
304
|
-
break;
|
|
305
|
-
}
|
|
306
|
-
}
|
|
307
|
-
if (!response) {
|
|
308
|
-
response = await requestHandler(request, serverContext);
|
|
309
|
-
}
|
|
310
|
-
for (const onResponseHook of onResponseHooks) {
|
|
311
|
-
await onResponseHook({
|
|
312
|
-
request,
|
|
313
|
-
response,
|
|
314
|
-
serverContext,
|
|
315
|
-
});
|
|
316
|
-
}
|
|
317
|
-
return response;
|
|
318
|
-
}
|
|
319
|
-
function handleNodeRequest(nodeRequest, ...ctx) {
|
|
320
|
-
const serverContext = ctx.length > 1 ? completeAssign({}, ...ctx) : ctx[0];
|
|
321
|
-
const request = normalizeNodeRequest(nodeRequest, fetchAPI.Request);
|
|
322
|
-
return handleRequest(request, serverContext);
|
|
323
|
-
}
|
|
324
|
-
async function requestListener(nodeRequest, serverResponse, ...ctx) {
|
|
325
|
-
const waitUntilPromises = [];
|
|
326
|
-
const defaultServerContext = {
|
|
327
|
-
req: nodeRequest,
|
|
328
|
-
res: serverResponse,
|
|
329
|
-
waitUntil(promise) {
|
|
330
|
-
if (promise != null) {
|
|
331
|
-
waitUntilPromises.push(promise);
|
|
332
|
-
}
|
|
333
|
-
},
|
|
334
|
-
};
|
|
335
|
-
const response = await handleNodeRequest(nodeRequest, defaultServerContext, ...ctx);
|
|
336
|
-
if (response) {
|
|
337
|
-
await sendNodeResponse(response, serverResponse, nodeRequest);
|
|
338
|
-
}
|
|
339
|
-
else {
|
|
340
|
-
await new Promise(resolve => {
|
|
341
|
-
serverResponse.statusCode = 404;
|
|
342
|
-
serverResponse.once('end', resolve);
|
|
343
|
-
serverResponse.end();
|
|
344
|
-
});
|
|
345
|
-
}
|
|
346
|
-
if (waitUntilPromises.length > 0) {
|
|
347
|
-
await handleWaitUntils(waitUntilPromises);
|
|
348
|
-
}
|
|
349
|
-
}
|
|
350
|
-
function handleEvent(event, ...ctx) {
|
|
351
|
-
if (!event.respondWith || !event.request) {
|
|
352
|
-
throw new TypeError(`Expected FetchEvent, got ${event}`);
|
|
353
|
-
}
|
|
354
|
-
const serverContext = ctx.length > 0 ? Object.assign({}, event, ...ctx) : event;
|
|
355
|
-
const response$ = handleRequest(event.request, serverContext);
|
|
356
|
-
event.respondWith(response$);
|
|
357
|
-
}
|
|
358
|
-
function handleRequestWithWaitUntil(request, ...ctx) {
|
|
359
|
-
const serverContext = ctx.length > 1 ? completeAssign({}, ...ctx) : ctx[0] || {};
|
|
360
|
-
if (!('waitUntil' in serverContext)) {
|
|
361
|
-
const waitUntilPromises = [];
|
|
362
|
-
const response$ = handleRequest(request, {
|
|
363
|
-
...serverContext,
|
|
364
|
-
waitUntil(promise) {
|
|
365
|
-
if (promise != null) {
|
|
366
|
-
waitUntilPromises.push(promise);
|
|
367
|
-
}
|
|
368
|
-
},
|
|
369
|
-
});
|
|
370
|
-
if (waitUntilPromises.length > 0) {
|
|
371
|
-
return handleWaitUntils(waitUntilPromises).then(() => response$);
|
|
372
|
-
}
|
|
373
|
-
return response$;
|
|
374
|
-
}
|
|
375
|
-
return handleRequest(request, serverContext);
|
|
376
|
-
}
|
|
377
|
-
const fetchFn = (input, ...maybeCtx) => {
|
|
378
|
-
if (typeof input === 'string' || 'href' in input) {
|
|
379
|
-
const [initOrCtx, ...restOfCtx] = maybeCtx;
|
|
380
|
-
if (isRequestInit(initOrCtx)) {
|
|
381
|
-
return handleRequestWithWaitUntil(new fetchAPI.Request(input, initOrCtx), ...restOfCtx);
|
|
382
|
-
}
|
|
383
|
-
return handleRequestWithWaitUntil(new fetchAPI.Request(input), ...maybeCtx);
|
|
384
|
-
}
|
|
385
|
-
return handleRequestWithWaitUntil(input, ...maybeCtx);
|
|
386
|
-
};
|
|
387
|
-
const genericRequestHandler = (input, ...maybeCtx) => {
|
|
388
|
-
// If it is a Node request
|
|
389
|
-
const [initOrCtxOrRes, ...restOfCtx] = maybeCtx;
|
|
390
|
-
if (isNodeRequest(input)) {
|
|
391
|
-
if (!isServerResponse(initOrCtxOrRes)) {
|
|
392
|
-
throw new TypeError(`Expected ServerResponse, got ${initOrCtxOrRes}`);
|
|
393
|
-
}
|
|
394
|
-
return requestListener(input, initOrCtxOrRes, ...restOfCtx);
|
|
395
|
-
}
|
|
396
|
-
if (isServerResponse(initOrCtxOrRes)) {
|
|
397
|
-
throw new TypeError('Got Node response without Node request');
|
|
398
|
-
}
|
|
399
|
-
// Is input a container object over Request?
|
|
400
|
-
if (isRequestAccessible(input)) {
|
|
401
|
-
// Is it FetchEvent?
|
|
402
|
-
if (isFetchEvent(input)) {
|
|
403
|
-
return handleEvent(input, ...maybeCtx);
|
|
404
|
-
}
|
|
405
|
-
// In this input is also the context
|
|
406
|
-
return handleRequestWithWaitUntil(input.request, input, ...maybeCtx);
|
|
407
|
-
}
|
|
408
|
-
// Or is it Request itself?
|
|
409
|
-
// Then ctx is present and it is the context
|
|
410
|
-
return fetchFn(input, ...maybeCtx);
|
|
411
|
-
};
|
|
412
|
-
const adapterObj = {
|
|
413
|
-
handleRequest,
|
|
414
|
-
fetch: fetchFn,
|
|
415
|
-
handleNodeRequest,
|
|
416
|
-
requestListener,
|
|
417
|
-
handleEvent,
|
|
418
|
-
handle: genericRequestHandler,
|
|
419
|
-
};
|
|
420
|
-
const serverAdapter = new Proxy(genericRequestHandler, {
|
|
421
|
-
// It should have all the attributes of the handler function and the server instance
|
|
422
|
-
has: (_, prop) => {
|
|
423
|
-
return (prop in adapterObj ||
|
|
424
|
-
prop in genericRequestHandler ||
|
|
425
|
-
(serverAdapterBaseObject && prop in serverAdapterBaseObject));
|
|
426
|
-
},
|
|
427
|
-
get: (_, prop) => {
|
|
428
|
-
const adapterProp = adapterObj[prop];
|
|
429
|
-
if (adapterProp) {
|
|
430
|
-
if (adapterProp.bind) {
|
|
431
|
-
return adapterProp.bind(adapterObj);
|
|
432
|
-
}
|
|
433
|
-
return adapterProp;
|
|
434
|
-
}
|
|
435
|
-
const handleProp = genericRequestHandler[prop];
|
|
436
|
-
if (handleProp) {
|
|
437
|
-
if (handleProp.bind) {
|
|
438
|
-
return handleProp.bind(genericRequestHandler);
|
|
439
|
-
}
|
|
440
|
-
return handleProp;
|
|
441
|
-
}
|
|
442
|
-
if (serverAdapterBaseObject) {
|
|
443
|
-
const serverAdapterBaseObjectProp = serverAdapterBaseObject[prop];
|
|
444
|
-
if (serverAdapterBaseObjectProp) {
|
|
445
|
-
if (serverAdapterBaseObjectProp.bind) {
|
|
446
|
-
return function (...args) {
|
|
447
|
-
const returnedVal = serverAdapterBaseObject[prop](...args);
|
|
448
|
-
if (returnedVal === serverAdapterBaseObject) {
|
|
449
|
-
return serverAdapter;
|
|
450
|
-
}
|
|
451
|
-
return returnedVal;
|
|
452
|
-
};
|
|
453
|
-
}
|
|
454
|
-
return serverAdapterBaseObjectProp;
|
|
455
|
-
}
|
|
456
|
-
}
|
|
457
|
-
},
|
|
458
|
-
apply(_, __, args) {
|
|
459
|
-
return genericRequestHandler(...args);
|
|
460
|
-
},
|
|
461
|
-
});
|
|
462
|
-
return serverAdapter;
|
|
463
|
-
}
|
|
464
|
-
// from https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/assign#copying_accessors
|
|
465
|
-
function completeAssign(target, ...sources) {
|
|
466
|
-
sources.forEach(source => {
|
|
467
|
-
if (source != null && typeof source === 'object') {
|
|
468
|
-
// modified Object.keys to Object.getOwnPropertyNames
|
|
469
|
-
// because Object.keys only returns enumerable properties
|
|
470
|
-
const descriptors = Object.getOwnPropertyNames(source).reduce((descriptors, key) => {
|
|
471
|
-
descriptors[key] = Object.getOwnPropertyDescriptor(source, key);
|
|
472
|
-
return descriptors;
|
|
473
|
-
}, {});
|
|
474
|
-
// By default, Object.assign copies enumerable Symbols, too
|
|
475
|
-
Object.getOwnPropertySymbols(source).forEach(sym => {
|
|
476
|
-
const descriptor = Object.getOwnPropertyDescriptor(source, sym);
|
|
477
|
-
if (descriptor.enumerable) {
|
|
478
|
-
descriptors[sym] = descriptor;
|
|
479
|
-
}
|
|
480
|
-
});
|
|
481
|
-
Object.defineProperties(target, descriptors);
|
|
482
|
-
}
|
|
483
|
-
});
|
|
484
|
-
return target;
|
|
485
|
-
}
|
|
486
|
-
|
|
487
|
-
function getCORSHeadersByRequestAndOptions(request, corsOptions) {
|
|
488
|
-
var _a, _b;
|
|
489
|
-
const currentOrigin = request.headers.get('origin');
|
|
490
|
-
if (corsOptions === false || currentOrigin == null) {
|
|
491
|
-
return null;
|
|
492
|
-
}
|
|
493
|
-
const headers = {};
|
|
494
|
-
// If defined origins have '*' or undefined by any means, we should allow all origins
|
|
495
|
-
if (corsOptions.origin == null ||
|
|
496
|
-
corsOptions.origin.length === 0 ||
|
|
497
|
-
corsOptions.origin.includes('*')) {
|
|
498
|
-
headers['Access-Control-Allow-Origin'] = currentOrigin;
|
|
499
|
-
// Vary by origin because there are multiple origins
|
|
500
|
-
headers['Vary'] = 'Origin';
|
|
501
|
-
}
|
|
502
|
-
else if (typeof corsOptions.origin === 'string') {
|
|
503
|
-
// If there is one specific origin is specified, use it directly
|
|
504
|
-
headers['Access-Control-Allow-Origin'] = corsOptions.origin;
|
|
505
|
-
}
|
|
506
|
-
else if (Array.isArray(corsOptions.origin)) {
|
|
507
|
-
// If there is only one origin defined in the array, consider it as a single one
|
|
508
|
-
if (corsOptions.origin.length === 1) {
|
|
509
|
-
headers['Access-Control-Allow-Origin'] = corsOptions.origin[0];
|
|
510
|
-
}
|
|
511
|
-
else if (corsOptions.origin.includes(currentOrigin)) {
|
|
512
|
-
// If origin is available in the headers, use it
|
|
513
|
-
headers['Access-Control-Allow-Origin'] = currentOrigin;
|
|
514
|
-
// Vary by origin because there are multiple origins
|
|
515
|
-
headers['Vary'] = 'Origin';
|
|
516
|
-
}
|
|
517
|
-
else {
|
|
518
|
-
// There is no origin found in the headers, so we should return null
|
|
519
|
-
headers['Access-Control-Allow-Origin'] = 'null';
|
|
520
|
-
}
|
|
521
|
-
}
|
|
522
|
-
if ((_a = corsOptions.methods) === null || _a === void 0 ? void 0 : _a.length) {
|
|
523
|
-
headers['Access-Control-Allow-Methods'] = corsOptions.methods.join(', ');
|
|
524
|
-
}
|
|
525
|
-
else {
|
|
526
|
-
const requestMethod = request.headers.get('access-control-request-method');
|
|
527
|
-
if (requestMethod) {
|
|
528
|
-
headers['Access-Control-Allow-Methods'] = requestMethod;
|
|
529
|
-
}
|
|
530
|
-
}
|
|
531
|
-
if ((_b = corsOptions.allowedHeaders) === null || _b === void 0 ? void 0 : _b.length) {
|
|
532
|
-
headers['Access-Control-Allow-Headers'] = corsOptions.allowedHeaders.join(', ');
|
|
533
|
-
}
|
|
534
|
-
else {
|
|
535
|
-
const requestHeaders = request.headers.get('access-control-request-headers');
|
|
536
|
-
if (requestHeaders) {
|
|
537
|
-
headers['Access-Control-Allow-Headers'] = requestHeaders;
|
|
538
|
-
if (headers['Vary']) {
|
|
539
|
-
headers['Vary'] += ', Access-Control-Request-Headers';
|
|
540
|
-
}
|
|
541
|
-
headers['Vary'] = 'Access-Control-Request-Headers';
|
|
542
|
-
}
|
|
543
|
-
}
|
|
544
|
-
if (corsOptions.credentials != null) {
|
|
545
|
-
if (corsOptions.credentials === true) {
|
|
546
|
-
headers['Access-Control-Allow-Credentials'] = 'true';
|
|
547
|
-
}
|
|
548
|
-
}
|
|
549
|
-
else if (headers['Access-Control-Allow-Origin'] !== '*') {
|
|
550
|
-
headers['Access-Control-Allow-Credentials'] = 'true';
|
|
551
|
-
}
|
|
552
|
-
if (corsOptions.exposedHeaders) {
|
|
553
|
-
headers['Access-Control-Expose-Headers'] = corsOptions.exposedHeaders.join(', ');
|
|
554
|
-
}
|
|
555
|
-
if (corsOptions.maxAge) {
|
|
556
|
-
headers['Access-Control-Max-Age'] = corsOptions.maxAge.toString();
|
|
557
|
-
}
|
|
558
|
-
return headers;
|
|
559
|
-
}
|
|
560
|
-
async function getCORSResponseHeaders(request, corsOptionsFactory, serverContext) {
|
|
561
|
-
const corsOptions = await corsOptionsFactory(request, serverContext);
|
|
562
|
-
return getCORSHeadersByRequestAndOptions(request, corsOptions);
|
|
563
|
-
}
|
|
564
|
-
function useCORS(options) {
|
|
565
|
-
let corsOptionsFactory = () => ({});
|
|
566
|
-
if (options != null) {
|
|
567
|
-
if (typeof options === 'function') {
|
|
568
|
-
corsOptionsFactory = options;
|
|
569
|
-
}
|
|
570
|
-
else if (typeof options === 'object') {
|
|
571
|
-
const corsOptions = {
|
|
572
|
-
...options,
|
|
573
|
-
};
|
|
574
|
-
corsOptionsFactory = () => corsOptions;
|
|
575
|
-
}
|
|
576
|
-
else if (options === false) {
|
|
577
|
-
corsOptionsFactory = () => false;
|
|
578
|
-
}
|
|
579
|
-
}
|
|
580
|
-
return {
|
|
581
|
-
onRequest({ request, fetchAPI, endResponse }) {
|
|
582
|
-
if (request.method.toUpperCase() === 'OPTIONS') {
|
|
583
|
-
const response = new fetchAPI.Response(null, {
|
|
584
|
-
status: 204,
|
|
585
|
-
// Safari (and potentially other browsers) need content-length 0,
|
|
586
|
-
// for 204 or they just hang waiting for a body
|
|
587
|
-
// see: https://github.com/expressjs/cors/blob/master/lib/index.js#L176
|
|
588
|
-
headers: {
|
|
589
|
-
'Content-Length': '0',
|
|
590
|
-
},
|
|
591
|
-
});
|
|
592
|
-
endResponse(response);
|
|
593
|
-
}
|
|
594
|
-
},
|
|
595
|
-
async onResponse({ request, serverContext, response }) {
|
|
596
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
597
|
-
const headers = await getCORSResponseHeaders(request, corsOptionsFactory, serverContext);
|
|
598
|
-
if (headers != null) {
|
|
599
|
-
for (const headerName in headers) {
|
|
600
|
-
response.headers.set(headerName, headers[headerName]);
|
|
601
|
-
}
|
|
602
|
-
}
|
|
603
|
-
},
|
|
604
|
-
};
|
|
605
|
-
}
|
|
606
|
-
|
|
607
|
-
function createDefaultErrorHandler(ResponseCtor = DefaultFetchAPI.Response) {
|
|
608
|
-
return function defaultErrorHandler(e) {
|
|
609
|
-
return new ResponseCtor(typeof e.details === 'object'
|
|
610
|
-
? JSON.stringify(e.details)
|
|
611
|
-
: e.stack || e.message || e.toString(), {
|
|
612
|
-
status: e.statusCode || e.status || 500,
|
|
613
|
-
headers: e.headers || {},
|
|
614
|
-
});
|
|
615
|
-
};
|
|
616
|
-
}
|
|
617
|
-
class HTTPError extends Error {
|
|
618
|
-
constructor(status, message, headers = {}, details) {
|
|
619
|
-
super(message);
|
|
620
|
-
this.status = status;
|
|
621
|
-
this.message = message;
|
|
622
|
-
this.headers = headers;
|
|
623
|
-
this.details = details;
|
|
624
|
-
Error.captureStackTrace(this, HTTPError);
|
|
625
|
-
}
|
|
626
|
-
}
|
|
627
|
-
function useErrorHandling(onError) {
|
|
628
|
-
return {
|
|
629
|
-
onRequest({ requestHandler, setRequestHandler, fetchAPI }) {
|
|
630
|
-
const errorHandler = onError || createDefaultErrorHandler(fetchAPI.Response);
|
|
631
|
-
setRequestHandler(async function handlerWithErrorHandling(request, serverContext) {
|
|
632
|
-
try {
|
|
633
|
-
const response = await requestHandler(request, serverContext);
|
|
634
|
-
return response;
|
|
635
|
-
}
|
|
636
|
-
catch (e) {
|
|
637
|
-
const response = await errorHandler(e, request, serverContext);
|
|
638
|
-
return response;
|
|
639
|
-
}
|
|
640
|
-
});
|
|
641
|
-
},
|
|
642
|
-
};
|
|
643
|
-
}
|
|
644
|
-
|
|
645
|
-
Object.defineProperty(exports, 'Response', {
|
|
646
|
-
enumerable: true,
|
|
647
|
-
get: function () {
|
|
648
|
-
return DefaultFetchAPI.Response;
|
|
649
|
-
}
|
|
650
|
-
});
|
|
651
|
-
exports.HTTPError = HTTPError;
|
|
652
|
-
exports.createDefaultErrorHandler = createDefaultErrorHandler;
|
|
653
|
-
exports.createServerAdapter = createServerAdapter;
|
|
654
|
-
exports.getCORSHeadersByRequestAndOptions = getCORSHeadersByRequestAndOptions;
|
|
655
|
-
exports.getHeadersObj = getHeadersObj;
|
|
656
|
-
exports.isAsyncIterable = isAsyncIterable;
|
|
657
|
-
exports.isFetchEvent = isFetchEvent;
|
|
658
|
-
exports.isNodeRequest = isNodeRequest;
|
|
659
|
-
exports.isReadable = isReadable;
|
|
660
|
-
exports.isReadableStream = isReadableStream;
|
|
661
|
-
exports.isRequestInit = isRequestInit;
|
|
662
|
-
exports.isServerResponse = isServerResponse;
|
|
663
|
-
exports.normalizeNodeRequest = normalizeNodeRequest;
|
|
664
|
-
exports.sendNodeResponse = sendNodeResponse;
|
|
665
|
-
exports.useCORS = useCORS;
|
|
666
|
-
exports.useErrorHandling = useErrorHandling;
|