1
1
import type { OutgoingHttpHeaders } from 'http'
2
2
3
3
import { ComputeJsOutgoingMessage , toComputeResponse , toReqRes } from '@fastly/http-compute-js'
4
- import { Context } from '@netlify/functions'
5
4
import type { NextConfigComplete } from 'next/dist/server/config-shared.js'
6
5
import type { WorkerRequestHandler } from 'next/dist/server/lib/types.js'
7
6
@@ -14,12 +13,19 @@ import {
14
13
} from '../headers.js'
15
14
import { nextResponseProxy } from '../revalidate.js'
16
15
17
- import { createRequestContext , getLogger , getRequestContext } from './request-context.cjs'
16
+ import {
17
+ createRequestContext ,
18
+ FutureContext ,
19
+ getLogger ,
20
+ getRequestContext ,
21
+ } from './request-context.cjs'
18
22
import { getTracer } from './tracer.cjs'
19
- import { setWaitUntil } from './wait-until.cjs'
23
+ import { setupWaitUntil } from './wait-until.cjs'
20
24
21
25
const nextImportPromise = import ( '../next.cjs' )
22
26
27
+ setupWaitUntil ( )
28
+
23
29
let nextHandler : WorkerRequestHandler , nextConfig : NextConfigComplete
24
30
25
31
/**
@@ -45,15 +51,9 @@ const disableFaultyTransferEncodingHandling = (res: ComputeJsOutgoingMessage) =>
45
51
}
46
52
}
47
53
48
- // TODO: remove once https://github.com/netlify/serverless-functions-api/pull/219
49
- // is released and public types are updated
50
- interface FutureContext extends Context {
51
- waitUntil ?: ( promise : Promise < unknown > ) => void
52
- }
53
-
54
54
export default async ( request : Request , context : FutureContext ) => {
55
55
const tracer = getTracer ( )
56
- setWaitUntil ( context )
56
+
57
57
if ( ! nextHandler ) {
58
58
await tracer . withActiveSpan ( 'initialize next server' , async ( ) => {
59
59
// set the server config
@@ -129,19 +129,19 @@ export default async (request: Request, context: FutureContext) => {
129
129
return new Response ( body || null , response )
130
130
}
131
131
132
- if ( context . waitUntil ) {
133
- context . waitUntil ( requestContext . backgroundWorkPromise )
134
- }
135
-
136
132
const keepOpenUntilNextFullyRendered = new TransformStream ( {
137
133
async flush ( ) {
138
134
// it's important to keep the stream open until the next handler has finished
139
135
await nextHandlerPromise
140
- if ( ! context . waitUntil ) {
141
- // if waitUntil is not available, we have to keep response stream open until background promises are resolved
142
- // to ensure that all background work executes
143
- await requestContext . backgroundWorkPromise
144
- }
136
+
137
+ // Next.js relies on `close` event emitted by response to trigger running callback variant of `next/after`
138
+ // however @fastly /http-compute-js never actually emits that event - so we have to emit it ourselves, otherwise
139
+ // Next would never run the callback variant of `next/after`
140
+ res . emit ( 'close' )
141
+
142
+ // if waitUntil is not available, we have to keep response stream open until background promises are resolved
143
+ // to ensure that all background work executes
144
+ await requestContext . backgroundWorkPromise
145
145
} ,
146
146
} )
147
147
0 commit comments