-
Notifications
You must be signed in to change notification settings - Fork 149
/
Copy pathrequestHandler.ts
241 lines (217 loc) · 7.81 KB
/
requestHandler.ts
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
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
import { AsyncLocalStorage } from "node:async_hooks";
import type { OpenNextNodeResponse, StreamCreator } from "http/index.js";
import { IncomingMessage } from "http/index.js";
import type { InternalEvent, InternalResult } from "types/open-next";
import { runWithOpenNextRequestContext } from "utils/promise";
import { debug, error, warn } from "../adapters/logger";
import { patchAsyncStorage } from "./patchAsyncStorage";
import { resolveProxyRequest } from "./resolve";
import { convertRes, createServerResponse } from "./routing/util";
import type { MiddlewareOutputEvent } from "./routingHandler";
import routingHandler, {
MIDDLEWARE_HEADER_PREFIX,
MIDDLEWARE_HEADER_PREFIX_LEN,
} from "./routingHandler";
import { requestHandler, setNextjsPrebundledReact } from "./util";
// This is used to identify requests in the cache
globalThis.__openNextAls = new AsyncLocalStorage();
//#override patchAsyncStorage
patchAsyncStorage();
//#endOverride
export async function openNextHandler(
internalEvent: InternalEvent,
responseStreaming?: StreamCreator,
): Promise<InternalResult> {
// We run everything in the async local storage context so that it is available in the middleware as well as in NextServer
return runWithOpenNextRequestContext(
{ isISRRevalidation: internalEvent.headers["x-isr"] === "1" },
async () => {
if (internalEvent.headers["x-forwarded-host"]) {
internalEvent.headers.host = internalEvent.headers["x-forwarded-host"];
}
debug("internalEvent", internalEvent);
let routingResult: InternalResult | MiddlewareOutputEvent = {
type: "middleware",
internalEvent,
isExternalRewrite: false,
origin: false,
isISR: false,
};
//#override withRouting
try {
routingResult = await routingHandler(internalEvent);
} catch (e) {
warn("Routing failed.", e);
}
//#endOverride
const headers =
routingResult.type === "middleware"
? routingResult.internalEvent.headers
: routingResult.headers;
const overwrittenResponseHeaders: Record<string, string | string[]> = {};
for (const [rawKey, value] of Object.entries(headers)) {
if (!rawKey.startsWith(MIDDLEWARE_HEADER_PREFIX)) {
continue;
}
const key = rawKey.slice(MIDDLEWARE_HEADER_PREFIX_LEN);
overwrittenResponseHeaders[key] = value;
headers[key] = value;
delete headers[rawKey];
}
if (
routingResult.type === "middleware" &&
routingResult.isExternalRewrite === true
) {
try {
routingResult = await globalThis.proxyExternalRequest.proxy(
routingResult.internalEvent,
);
} catch (e) {
error("External request failed.", e);
routingResult = {
type: "middleware",
internalEvent: {
type: "core",
rawPath: "/500",
method: "GET",
headers: {},
url: "/500",
query: {},
cookies: {},
remoteAddress: "",
},
// On error we need to rewrite to the 500 page which is an internal rewrite
isExternalRewrite: false,
isISR: false,
origin: false,
};
}
}
if (routingResult.type === "core") {
// response is used only in the streaming case
if (responseStreaming) {
const response = createServerResponse(
internalEvent,
headers,
responseStreaming,
);
response.statusCode = routingResult.statusCode;
response.flushHeaders();
const [bodyToConsume, bodyToReturn] = routingResult.body.tee();
for await (const chunk of bodyToConsume) {
response.write(chunk);
}
response.end();
routingResult.body = bodyToReturn;
}
return routingResult;
}
const preprocessedEvent = routingResult.internalEvent;
debug("preprocessedEvent", preprocessedEvent);
const reqProps = {
method: preprocessedEvent.method,
url: preprocessedEvent.url,
//WORKAROUND: We pass this header to the serverless function to mimic a prefetch request which will not trigger revalidation since we handle revalidation differently
// There is 3 way we can handle revalidation:
// 1. We could just let the revalidation go as normal, but due to race condtions the revalidation will be unreliable
// 2. We could alter the lastModified time of our cache to make next believe that the cache is fresh, but this could cause issues with stale data since the cdn will cache the stale data as if it was fresh
// 3. OUR CHOICE: We could pass a purpose prefetch header to the serverless function to make next believe that the request is a prefetch request and not trigger revalidation (This could potentially break in the future if next changes the behavior of prefetch requests)
headers: { ...headers, purpose: "prefetch" },
body: preprocessedEvent.body,
remoteAddress: preprocessedEvent.remoteAddress,
};
const mergeHeadersPriority = globalThis.openNextConfig.dangerous
?.headersAndCookiesPriority
? globalThis.openNextConfig.dangerous.headersAndCookiesPriority(
preprocessedEvent,
)
: "middleware";
const store = globalThis.__openNextAls.getStore();
if (store) {
store.mergeHeadersPriority = mergeHeadersPriority;
}
const req = new IncomingMessage(reqProps);
const res = createServerResponse(
preprocessedEvent,
overwrittenResponseHeaders,
responseStreaming,
);
await processRequest(req, res, preprocessedEvent);
const {
statusCode,
headers: responseHeaders,
isBase64Encoded,
body,
} = convertRes(res);
const internalResult = {
type: internalEvent.type,
statusCode,
headers: responseHeaders,
body,
isBase64Encoded,
};
const requestId = store?.requestId;
if (requestId) {
// reset lastModified. We need to do this to avoid memory leaks
delete globalThis.lastModified[requestId];
}
return internalResult;
},
);
}
async function processRequest(
req: IncomingMessage,
res: OpenNextNodeResponse,
internalEvent: InternalEvent,
) {
// @ts-ignore
// Next.js doesn't parse body if the property exists
// https://github.com/dougmoscrop/serverless-http/issues/227
delete req.body;
try {
//#override applyNextjsPrebundledReact
setNextjsPrebundledReact(internalEvent.rawPath);
//#endOverride
// Next Server
await requestHandler(req, res);
} catch (e: any) {
// This might fail when using bundled next, importing won't do the trick either
if (e.constructor.name === "NoFallbackError") {
// Do we need to handle _not-found
// Ideally this should never get triggered and be intercepted by the routing handler
await tryRenderError("404", res, internalEvent);
} else {
error("NextJS request failed.", e);
await tryRenderError("500", res, internalEvent);
}
}
}
async function tryRenderError(
type: "404" | "500",
res: OpenNextNodeResponse,
internalEvent: InternalEvent,
) {
try {
const _req = new IncomingMessage({
method: "GET",
url: `/${type}`,
headers: internalEvent.headers,
body: internalEvent.body,
remoteAddress: internalEvent.remoteAddress,
});
await requestHandler(_req, res);
} catch (e) {
error("NextJS request failed.", e);
res.setHeader("Content-Type", "application/json");
res.end(
JSON.stringify(
{
message: "Server failed to respond.",
details: e,
},
null,
2,
),
);
}
}