Skip to content

Commit 51e5373

Browse files
piehascorbic
andauthored
feat: support after() (#2717)
* test: upgrade deno used in tests * test: e2e simple next/after test * feat: support waitUntil * use already existing trackBackgroundWork helper from request context to handle next/after * test: increase timeout for one of smoke tests due to team-wide extensions installation time making it timeout * test: move test to dedicated fixture * chore: update outdated comment * chore: clarify awaiting backgroundWorkPromise * test: update assertion for next >=15.0.4-canary.18 * fix: support changed shape of getRequestHandlers return --------- Co-authored-by: Matt Kane <[email protected]>
1 parent 00e3a4b commit 51e5373

20 files changed

+208
-44
lines changed

.github/workflows/pre-release.yml

+2-2
Original file line numberDiff line numberDiff line change
@@ -23,8 +23,8 @@ jobs:
2323
- name: Install Deno
2424
uses: denoland/setup-deno@v1
2525
with:
26-
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/edge-bundler/blob/e55f825bd985d3c92e21d1b765d71e70d5628fba/node/bridge.ts#L17
27-
deno-version: v1.37.0
26+
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/build/blob/main/packages/edge-bundler/node/bridge.ts#L20
27+
deno-version: v1.44.4
2828
- name: Extract tag and version
2929
id: extract
3030
run: |-

.github/workflows/release-please.yml

+2-2
Original file line numberDiff line numberDiff line change
@@ -31,8 +31,8 @@ jobs:
3131
- name: Install Deno
3232
uses: denoland/setup-deno@v1
3333
with:
34-
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/edge-bundler/blob/e55f825bd985d3c92e21d1b765d71e70d5628fba/node/bridge.ts#L17
35-
deno-version: v1.37.0
34+
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/build/blob/main/packages/edge-bundler/node/bridge.ts#L20
35+
deno-version: v1.44.4
3636
- name: Build
3737
run: npm run build
3838
if: ${{ steps.release.outputs.release_created }}

.github/workflows/run-tests.yml

+5-5
Original file line numberDiff line numberDiff line change
@@ -65,8 +65,8 @@ jobs:
6565
- name: Install Deno
6666
uses: denoland/setup-deno@v1
6767
with:
68-
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/edge-bundler/blob/main/node/bridge.ts#L17
69-
deno-version: v1.37.0
68+
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/build/blob/main/packages/edge-bundler/node/bridge.ts#L20
69+
deno-version: v1.44.4
7070
- name: 'Install dependencies'
7171
run: npm ci
7272
- name: 'Prepare Netlify CLI'
@@ -134,7 +134,7 @@ jobs:
134134
uses: denoland/setup-deno@v1
135135
with:
136136
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/edge-bundler/blob/e55f825bd985d3c92e21d1b765d71e70d5628fba/node/bridge.ts#L17
137-
deno-version: v1.37.0
137+
deno-version: v1.44.4
138138
- name: 'Install dependencies'
139139
run: npm ci
140140
- name: 'Build'
@@ -198,8 +198,8 @@ jobs:
198198
- name: Install Deno
199199
uses: denoland/setup-deno@v1
200200
with:
201-
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/edge-bundler/blob/main/node/bridge.ts#L17
202-
deno-version: v1.37.0
201+
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/build/blob/main/packages/edge-bundler/node/bridge.ts#L20
202+
deno-version: v1.44.4
203203
- name: 'Install dependencies'
204204
run: npm ci
205205
- name: 'Build'

.github/workflows/size-check.yml

+2-2
Original file line numberDiff line numberDiff line change
@@ -23,8 +23,8 @@ jobs:
2323
- name: Install Deno
2424
uses: denoland/setup-deno@v1
2525
with:
26-
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/edge-bundler/blob/main/node/bridge.ts#L17
27-
deno-version: v1.37.0
26+
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/build/blob/main/packages/edge-bundler/node/bridge.ts#L20
27+
deno-version: v1.44.4
2828
- run: npm ci
2929

3030
- name: Package size report

.github/workflows/test-e2e.yml

+2-2
Original file line numberDiff line numberDiff line change
@@ -161,8 +161,8 @@ jobs:
161161
- name: Install Deno
162162
uses: denoland/setup-deno@v1
163163
with:
164-
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/edge-bundler/blob/main/node/bridge.ts#L17
165-
deno-version: v1.37.0
164+
# Should match the `DENO_VERSION_RANGE` from https://github.com/netlify/build/blob/main/packages/edge-bundler/node/bridge.ts#L20
165+
deno-version: v1.44.4
166166

167167
- name: install runtime
168168
run: npm install --ignore-scripts

src/build/templates/handler-monorepo.tmpl.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ export default async function (req, context) {
1616
tracing.start()
1717
}
1818

19-
const requestContext = createRequestContext(req)
19+
const requestContext = createRequestContext(req, context)
2020
const tracer = getTracer()
2121

2222
const handlerResponse = await runWithRequestContext(requestContext, () => {

src/build/templates/handler.tmpl.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ export default async function handler(req, context) {
1313
if (process.env.NETLIFY_OTLP_TRACE_EXPORTER_URL) {
1414
tracing.start()
1515
}
16-
const requestContext = createRequestContext(req)
16+
const requestContext = createRequestContext(req, context)
1717
const tracer = getTracer()
1818

1919
const handlerResponse = await runWithRequestContext(requestContext, () => {

src/run/handlers/request-context.cts

+18-4
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,17 @@
11
import { AsyncLocalStorage } from 'node:async_hooks'
22

3+
import type { Context } from '@netlify/functions'
34
import { LogLevel, systemLogger } from '@netlify/functions/internal'
45

56
import type { NetlifyCachedRouteValue } from '../../shared/cache-types.cjs'
67

78
type SystemLogger = typeof systemLogger
89

10+
// TODO: remove once public types are updated
11+
export interface FutureContext extends Context {
12+
waitUntil?: (promise: Promise<unknown>) => void
13+
}
14+
915
export type RequestContext = {
1016
captureServerTiming: boolean
1117
responseCacheGetLastModified?: number
@@ -16,25 +22,33 @@ export type RequestContext = {
1622
serverTiming?: string
1723
routeHandlerRevalidate?: NetlifyCachedRouteValue['revalidate']
1824
/**
19-
* Track promise running in the background and need to be waited for
25+
* Track promise running in the background and need to be waited for.
26+
* Uses `context.waitUntil` if available, otherwise stores promises to
27+
* await on.
2028
*/
2129
trackBackgroundWork: (promise: Promise<unknown>) => void
2230
/**
23-
* Promise that need to be executed even if response was already sent
31+
* Promise that need to be executed even if response was already sent.
32+
* If `context.waitUntil` is available this promise will be always resolved
33+
* because background work tracking was offloaded to `context.waitUntil`.
2434
*/
2535
backgroundWorkPromise: Promise<unknown>
2636
logger: SystemLogger
2737
}
2838

2939
type RequestContextAsyncLocalStorage = AsyncLocalStorage<RequestContext>
3040

31-
export function createRequestContext(request?: Request): RequestContext {
41+
export function createRequestContext(request?: Request, context?: FutureContext): RequestContext {
3242
const backgroundWorkPromises: Promise<unknown>[] = []
3343

3444
return {
3545
captureServerTiming: request?.headers.has('x-next-debug-logging') ?? false,
3646
trackBackgroundWork: (promise) => {
37-
backgroundWorkPromises.push(promise)
47+
if (context?.waitUntil) {
48+
context.waitUntil(promise)
49+
} else {
50+
backgroundWorkPromises.push(promise)
51+
}
3852
},
3953
get backgroundWorkPromise() {
4054
return Promise.allSettled(backgroundWorkPromises)

src/run/handlers/server.ts

+16-19
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
import type { OutgoingHttpHeaders } from 'http'
22

33
import { ComputeJsOutgoingMessage, toComputeResponse, toReqRes } from '@fastly/http-compute-js'
4-
import { Context } from '@netlify/functions'
54
import type { NextConfigComplete } from 'next/dist/server/config-shared.js'
65
import type { WorkerRequestHandler } from 'next/dist/server/lib/types.js'
76

@@ -16,9 +15,12 @@ import { nextResponseProxy } from '../revalidate.js'
1615

1716
import { createRequestContext, getLogger, getRequestContext } from './request-context.cjs'
1817
import { getTracer } from './tracer.cjs'
18+
import { setupWaitUntil } from './wait-until.cjs'
1919

2020
const nextImportPromise = import('../next.cjs')
2121

22+
setupWaitUntil()
23+
2224
let nextHandler: WorkerRequestHandler, nextConfig: NextConfigComplete
2325

2426
/**
@@ -44,13 +46,7 @@ const disableFaultyTransferEncodingHandling = (res: ComputeJsOutgoingMessage) =>
4446
}
4547
}
4648

47-
// TODO: remove once https://github.com/netlify/serverless-functions-api/pull/219
48-
// is released and public types are updated
49-
interface FutureContext extends Context {
50-
waitUntil?: (promise: Promise<unknown>) => void
51-
}
52-
53-
export default async (request: Request, context: FutureContext) => {
49+
export default async (request: Request) => {
5450
const tracer = getTracer()
5551

5652
if (!nextHandler) {
@@ -60,10 +56,10 @@ export default async (request: Request, context: FutureContext) => {
6056
nextConfig = await getRunConfig()
6157
setRunConfig(nextConfig)
6258

63-
const { getMockedRequestHandlers } = await nextImportPromise
59+
const { getMockedRequestHandler } = await nextImportPromise
6460
const url = new URL(request.url)
6561

66-
;[nextHandler] = await getMockedRequestHandlers({
62+
nextHandler = await getMockedRequestHandler({
6763
port: Number(url.port) || 443,
6864
hostname: url.hostname,
6965
dir: process.cwd(),
@@ -128,19 +124,20 @@ export default async (request: Request, context: FutureContext) => {
128124
return new Response(body || null, response)
129125
}
130126

131-
if (context.waitUntil) {
132-
context.waitUntil(requestContext.backgroundWorkPromise)
133-
}
134-
135127
const keepOpenUntilNextFullyRendered = new TransformStream({
136128
async flush() {
137129
// it's important to keep the stream open until the next handler has finished
138130
await nextHandlerPromise
139-
if (!context.waitUntil) {
140-
// if waitUntil is not available, we have to keep response stream open until background promises are resolved
141-
// to ensure that all background work executes
142-
await requestContext.backgroundWorkPromise
143-
}
131+
132+
// Next.js relies on `close` event emitted by response to trigger running callback variant of `next/after`
133+
// however @fastly/http-compute-js never actually emits that event - so we have to emit it ourselves,
134+
// otherwise Next would never run the callback variant of `next/after`
135+
res.emit('close')
136+
137+
// We have to keep response stream open until tracked background promises that are don't use `context.waitUntil`
138+
// are resolved. If `context.waitUntil` is available, `requestContext.backgroundWorkPromise` will be empty
139+
// resolved promised and so awaiting it is no-op
140+
await requestContext.backgroundWorkPromise
144141
},
145142
})
146143

src/run/handlers/wait-until.cts

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
import { getRequestContext } from './request-context.cjs'
2+
3+
/**
4+
* @see https://github.com/vercel/next.js/blob/canary/packages/next/src/server/after/builtin-request-context.ts
5+
*/
6+
const NEXT_REQUEST_CONTEXT_SYMBOL = Symbol.for('@next/request-context')
7+
8+
export type NextJsRequestContext = {
9+
get(): { waitUntil?: (promise: Promise<unknown>) => void } | undefined
10+
}
11+
12+
type GlobalThisWithRequestContext = typeof globalThis & {
13+
[NEXT_REQUEST_CONTEXT_SYMBOL]?: NextJsRequestContext
14+
}
15+
16+
/**
17+
* Registers a `waitUntil` to be used by Next.js for next/after
18+
*/
19+
export function setupWaitUntil() {
20+
// eslint-disable-next-line @typescript-eslint/no-extra-semi
21+
;(globalThis as GlobalThisWithRequestContext)[NEXT_REQUEST_CONTEXT_SYMBOL] = {
22+
get() {
23+
return { waitUntil: getRequestContext()?.trackBackgroundWork }
24+
},
25+
}
26+
}

src/run/next.cts

+5-2
Original file line numberDiff line numberDiff line change
@@ -85,7 +85,7 @@ export type HtmlBlob = {
8585
isFallback: boolean
8686
}
8787

88-
export async function getMockedRequestHandlers(...args: Parameters<typeof getRequestHandlers>) {
88+
export async function getMockedRequestHandler(...args: Parameters<typeof getRequestHandlers>) {
8989
const tracer = getTracer()
9090
return tracer.withActiveSpan('mocked request handler', async () => {
9191
const ofs = { ...fs }
@@ -131,6 +131,9 @@ export async function getMockedRequestHandlers(...args: Parameters<typeof getReq
131131
require('fs').promises,
132132
)
133133

134-
return getRequestHandlers(...args)
134+
const requestHandlers = await getRequestHandlers(...args)
135+
// depending on Next.js version requestHandlers might be an array of object
136+
// see https://github.com/vercel/next.js/commit/08e7410f15706379994b54c3195d674909a8d533#diff-37243d614f1f5d3f7ea50bbf2af263f6b1a9a4f70e84427977781e07b02f57f1R742
137+
return Array.isArray(requestHandlers) ? requestHandlers[0] : requestHandlers.requestHandler
135138
})
136139
}

tests/e2e/after.test.ts

+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
import { expect } from '@playwright/test'
2+
import { nextVersionSatisfies } from '../utils/next-version-helpers.mjs'
3+
import { test } from '../utils/playwright-helpers.js'
4+
5+
test('next/after callback is executed and finishes', async ({ page, after }) => {
6+
test.skip(!nextVersionSatisfies('>=15.0.0'), 'This test is only for Next.js 15+')
7+
8+
// trigger initial request to check page which might be stale and allow regenerating in background
9+
await page.goto(`${after.url}/after/check`)
10+
11+
await new Promise((resolve) => setTimeout(resolve, 5000))
12+
13+
// after it was possibly regenerated we can start checking actual content of the page
14+
await page.goto(`${after.url}/after/check`)
15+
const pageInfoLocator1 = await page.locator('#page-info')
16+
const pageInfo1 = JSON.parse((await pageInfoLocator1.textContent()) ?? '{}')
17+
18+
expect(typeof pageInfo1?.timestamp, 'Check page should have timestamp').toBe('number')
19+
20+
await page.goto(`${after.url}/after/check`)
21+
const pageInfoLocator2 = await page.locator('#page-info')
22+
const pageInfo2 = JSON.parse((await pageInfoLocator2.textContent()) ?? '{}')
23+
24+
expect(typeof pageInfo2?.timestamp, 'Check page should have timestamp').toBe('number')
25+
26+
expect(pageInfo2.timestamp, 'Check page should be cached').toBe(pageInfo1.timestamp)
27+
28+
await page.goto(`${after.url}/after/trigger`)
29+
30+
// wait for next/after to trigger revalidation of check page
31+
await new Promise((resolve) => setTimeout(resolve, 5000))
32+
33+
await page.goto(`${after.url}/after/check`)
34+
const pageInfoLocator3 = await page.locator('#page-info')
35+
const pageInfo3 = JSON.parse((await pageInfoLocator3.textContent()) ?? '{}')
36+
37+
expect(typeof pageInfo3?.timestamp, 'Check page should have timestamp').toBe('number')
38+
expect(
39+
pageInfo3.timestamp,
40+
'Check page should be invalidated with newer timestamp',
41+
).toBeGreaterThan(pageInfo1.timestamp)
42+
})
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
export const revalidate = 3600 // arbitrarily long, just so that it doesn't happen during a test run
2+
3+
export default async function Page() {
4+
const data = {
5+
timestamp: Date.now(),
6+
}
7+
console.log('/timestamp/key/[key] rendered', data)
8+
9+
return <div id="page-info">{JSON.stringify(data)}</div>
10+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
import { revalidatePath } from 'next/cache'
2+
import { unstable_after as after, connection } from 'next/server'
3+
4+
export default async function Page() {
5+
await connection()
6+
after(async () => {
7+
// this will run after response was sent
8+
console.log('after() triggered')
9+
console.log('after() sleep 1s')
10+
await new Promise((resolve) => setTimeout(resolve, 1000))
11+
12+
console.log('after() revalidatePath /after/check')
13+
revalidatePath('/after/check')
14+
})
15+
16+
return <div>Page with after()</div>
17+
}

tests/fixtures/after/app/layout.js

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
export const metadata = {
2+
title: 'Simple Next App',
3+
description: 'Description for Simple Next App',
4+
}
5+
6+
export default function RootLayout({ children }) {
7+
return (
8+
<html lang="en">
9+
<body>{children}</body>
10+
</html>
11+
)
12+
}

tests/fixtures/after/next.config.js

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
/** @type {import('next').NextConfig} */
2+
const nextConfig = {
3+
output: 'standalone',
4+
eslint: {
5+
ignoreDuringBuilds: true,
6+
},
7+
experimental: {
8+
after: true,
9+
},
10+
}
11+
12+
module.exports = nextConfig

tests/fixtures/after/package.json

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
{
2+
"name": "after",
3+
"version": "0.1.0",
4+
"private": true,
5+
"scripts": {
6+
"postinstall": "next build",
7+
"dev": "next dev",
8+
"build": "next build"
9+
},
10+
"dependencies": {
11+
"next": "latest",
12+
"react": "18.2.0",
13+
"react-dom": "18.2.0"
14+
},
15+
"test": {
16+
"dependencies": {
17+
"next": ">=15.0.0"
18+
}
19+
}
20+
}

0 commit comments

Comments
 (0)