Skip to content

Commit 9c0490c

Browse files
authored
fix: handle ipx redirect that visitors might have browser cached from v4 (#390)
* test: add setup replicating runtime v4 next/image handling to test handling of potentially browser-cached v4 image handling redirects * fix: handle ipx redirect that visitors might have browser cached from v4
1 parent 98eb35f commit 9c0490c

File tree

6 files changed

+97
-15
lines changed

6 files changed

+97
-15
lines changed

src/build/image-cdn.ts

+21-9
Original file line numberDiff line numberDiff line change
@@ -18,15 +18,27 @@ export const setImageConfig = async (ctx: PluginContext): Promise<void> => {
1818
return
1919
}
2020

21-
ctx.netlifyConfig.redirects.push({
22-
from: imageEndpointPath,
23-
// w and q are too short to be used as params with id-length rule
24-
// but we are forced to do so because of the next/image loader decides on their names
25-
// eslint-disable-next-line id-length
26-
query: { url: ':url', w: ':width', q: ':quality' },
27-
to: '/.netlify/images?url=:url&w=:width&q=:quality',
28-
status: 200,
29-
})
21+
ctx.netlifyConfig.redirects.push(
22+
{
23+
from: imageEndpointPath,
24+
// w and q are too short to be used as params with id-length rule
25+
// but we are forced to do so because of the next/image loader decides on their names
26+
// eslint-disable-next-line id-length
27+
query: { url: ':url', w: ':width', q: ':quality' },
28+
to: '/.netlify/images?url=:url&w=:width&q=:quality',
29+
status: 200,
30+
},
31+
// when migrating from @netlify/plugin-nextjs@4 image redirect to ipx might be cached in the browser
32+
{
33+
from: '/_ipx/*',
34+
// w and q are too short to be used as params with id-length rule
35+
// but we are forced to do so because of the next/image loader decides on their names
36+
// eslint-disable-next-line id-length
37+
query: { url: ':url', w: ':width', q: ':quality' },
38+
to: '/.netlify/images?url=:url&w=:width&q=:quality',
39+
status: 200,
40+
},
41+
)
3042

3143
if (remotePatterns?.length !== 0 || domains?.length !== 0) {
3244
ctx.netlifyConfig.images ||= { remote_images: [] }

tests/e2e/simple-app.test.ts

+34-6
Original file line numberDiff line numberDiff line change
@@ -129,6 +129,8 @@ test.describe('next/image is using Netlify Image CDN', () => {
129129

130130
const nextImageResponse = await nextImageResponsePromise
131131
expect(nextImageResponse.request().url()).toContain('_next/image?url=%2Fsquirrel.jpg')
132+
133+
expect(nextImageResponse.status()).toBe(200)
132134
// ensure next/image is using Image CDN
133135
// source image is jpg, but when requesting it through Image CDN avif will be returned
134136
expect(await nextImageResponse.headerValue('content-type')).toEqual('image/avif')
@@ -152,8 +154,10 @@ test.describe('next/image is using Netlify Image CDN', () => {
152154
)}`,
153155
)
154156

155-
await expect(nextImageResponse?.status()).toBe(200)
156-
await expect(await nextImageResponse.headerValue('content-type')).toEqual('image/avif')
157+
expect(nextImageResponse.status()).toBe(200)
158+
expect(await nextImageResponse.headerValue('content-type')).toEqual('image/avif')
159+
160+
await expectImageWasLoaded(page.locator('img'))
157161
})
158162

159163
test('Remote images: remote patterns #2 (just hostname starting with wildcard)', async ({
@@ -172,8 +176,10 @@ test.describe('next/image is using Netlify Image CDN', () => {
172176
)}`,
173177
)
174178

175-
await expect(nextImageResponse?.status()).toBe(200)
176-
await expect(await nextImageResponse.headerValue('content-type')).toEqual('image/avif')
179+
expect(nextImageResponse.status()).toBe(200)
180+
expect(await nextImageResponse.headerValue('content-type')).toEqual('image/avif')
181+
182+
await expectImageWasLoaded(page.locator('img'))
177183
})
178184

179185
test('Remote images: domains', async ({ page, simpleNextApp }) => {
@@ -189,8 +195,30 @@ test.describe('next/image is using Netlify Image CDN', () => {
189195
)}`,
190196
)
191197

192-
await expect(nextImageResponse?.status()).toBe(200)
193-
await expect(await nextImageResponse.headerValue('content-type')).toEqual('image/avif')
198+
expect(nextImageResponse?.status()).toBe(200)
199+
expect(await nextImageResponse.headerValue('content-type')).toEqual('image/avif')
200+
201+
await expectImageWasLoaded(page.locator('img'))
202+
})
203+
204+
test('Handling of browser-cached Runtime v4 redirect', async ({ page, simpleNextApp }) => {
205+
// Runtime v4 redirects for next/image are 301 and would be cached by browser
206+
// So this test checks behavior when migrating from v4 to v5 for site visitors
207+
// and ensure that images are still served through Image CDN
208+
const nextImageResponsePromise = page.waitForResponse('**/_ipx/**')
209+
210+
await page.goto(`${simpleNextApp.url}/image/migration-from-v4-runtime`)
211+
212+
const nextImageResponse = await nextImageResponsePromise
213+
// ensure fixture is replicating runtime v4 redirect
214+
expect(nextImageResponse.request().url()).toContain(
215+
'_ipx/w_384,q_75/%2Fsquirrel.jpg?url=%2Fsquirrel.jpg&w=384&q=75',
216+
)
217+
218+
expect(nextImageResponse.status()).toEqual(200)
219+
expect(await nextImageResponse.headerValue('content-type')).toEqual('image/avif')
220+
221+
await expectImageWasLoaded(page.locator('img'))
194222
})
195223
})
196224

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
// Moved to separate component marked with "use client" to avoid the following error when attempting to do similar on page itself
2+
// Error: Functions cannot be passed directly to Client Components unless you explicitly expose it by marking it with "use server".
3+
// {0: ..., loader: function}
4+
5+
'use client'
6+
7+
import Image from 'next/image'
8+
9+
function RuntimeV4SimulatorImageLoader({ src, width, quality }) {
10+
// replicate default Next.js image loader, just using custom endpoint that will simulate the runtime V4 behavior
11+
// https://github.com/vercel/next.js/blob/c9439b5654432df6488e178e5ade6f4ad2d1cf6a/packages/next/src/shared/lib/image-loader.ts#L60
12+
return `/_nextRuntimeV4ImageHandler?url=${encodeURIComponent(src)}&w=${width}&q=${quality || 75}`
13+
}
14+
15+
export function NextImageWithLoaderSimulatingRuntimeV4(props) {
16+
return <Image {...props} loader={RuntimeV4SimulatorImageLoader} />
17+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
import { NextImageWithLoaderSimulatingRuntimeV4 } from './next-image-runtime-v4'
2+
3+
export default function NextImageUsingNetlifyImageCDN() {
4+
return (
5+
<main>
6+
<h1>Next/Image + Netlify Image CDN</h1>
7+
<NextImageWithLoaderSimulatingRuntimeV4
8+
src="/squirrel.jpg"
9+
alt="a cute squirrel (next/image)"
10+
width={300}
11+
height={278}
12+
/>
13+
</main>
14+
)
15+
}

tests/fixtures/simple-next-app/netlify.toml

+9
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,12 @@
1+
[[redirects]]
2+
from = "/_nextRuntimeV4ImageHandler"
3+
# replicate runtime V4 image handler
4+
# https://github.com/netlify/next-runtime/blob/637e08c3f3437e5e302ec230b8c849bb61495566/packages/runtime/src/helpers/functions.ts#L254-L259
5+
query = { url = ":url", w = ":width", q = ":quality" }
6+
to = "/_ipx/w_:width,q_:quality/:url"
7+
status = 301
8+
9+
110
[functions]
211
directory = "netlify/functions"
312
included_files = [

tests/integration/simple-app.test.ts

+1
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,7 @@ test<FixtureTestContext>('Test that the simple next app is working', async (ctx)
5858
expect(blobEntries.map(({ key }) => decodeBlobKey(key)).sort()).toEqual([
5959
'/404',
6060
'/image/local',
61+
'/image/migration-from-v4-runtime',
6162
'/image/remote-domain',
6263
'/image/remote-pattern-1',
6364
'/image/remote-pattern-2',

0 commit comments

Comments
 (0)