Skip to content

Commit 4bf8641

Browse files
piehtaty2010
andauthored
feat: handle /_next/image through Netlify Image CDN for local images (#149)
* image rewrite * check for default loader * add back image redirect * add nextconfig and netlifyconfig * tmp: revert back to using redirect for lambda until netlify/proxy#1402 is deployed to workaround redirect clash * refactor: move image-cdn handling to it's own module * chore: eslint ignore for short param names that we can't change * test: add netlifyConfig.redirects array to mocked plugin options * test: add unit test cases for image-cdn * test: add e2e test for image-cdn * Revert "tmp: revert back to using redirect for lambda until netlify/proxy#1402 is deployed to workaround redirect clash" This reverts commit bf3269db27cc0c9e0224bb2777e0f113d31d0741. * test: assert content-type instead of internal header that might change in e2e test * fix: use import type when importing types for safety --------- Co-authored-by: Tatyana <[email protected]>
1 parent ebe579f commit 4bf8641

File tree

8 files changed

+185
-3
lines changed

8 files changed

+185
-3
lines changed

src/build/image-cdn.test.ts

+117
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,117 @@
1+
/* eslint-disable id-length */
2+
import type { NetlifyPluginOptions } from '@netlify/build'
3+
import type { NextConfigComplete } from 'next/dist/server/config-shared.js'
4+
import { beforeEach, describe, expect, test, vi, TestContext } from 'vitest'
5+
6+
import { setImageConfig } from './image-cdn.js'
7+
import { PluginContext } from './plugin-context.js'
8+
9+
type DeepPartial<T> = T extends object
10+
? {
11+
[P in keyof T]?: DeepPartial<T[P]>
12+
}
13+
: T
14+
15+
type ImageCDNTestContext = TestContext & {
16+
pluginContext: PluginContext
17+
mockNextConfig?: DeepPartial<NextConfigComplete>
18+
}
19+
20+
describe('Image CDN', () => {
21+
beforeEach<ImageCDNTestContext>((ctx) => {
22+
ctx.mockNextConfig = undefined
23+
ctx.pluginContext = new PluginContext({
24+
netlifyConfig: {
25+
redirects: [],
26+
},
27+
} as unknown as NetlifyPluginOptions)
28+
vi.spyOn(ctx.pluginContext, 'getBuildConfig').mockImplementation(() =>
29+
Promise.resolve((ctx.mockNextConfig ?? {}) as NextConfigComplete),
30+
)
31+
})
32+
33+
test<ImageCDNTestContext>('adds redirect to Netlify Image CDN when default image loader is used', async (ctx) => {
34+
ctx.mockNextConfig = {
35+
images: {
36+
path: '/_next/image',
37+
loader: 'default',
38+
},
39+
}
40+
41+
await setImageConfig(ctx.pluginContext)
42+
43+
expect(ctx.pluginContext.netlifyConfig.redirects).toEqual(
44+
expect.arrayContaining([
45+
{
46+
from: '/_next/image',
47+
query: {
48+
q: ':quality',
49+
url: ':url',
50+
w: ':width',
51+
},
52+
to: '/.netlify/images?url=:url&w=:width&q=:quality',
53+
status: 200,
54+
},
55+
]),
56+
)
57+
})
58+
59+
test<ImageCDNTestContext>('does not add redirect to Netlify Image CDN when non-default loader is used', async (ctx) => {
60+
ctx.mockNextConfig = {
61+
images: {
62+
path: '/_next/image',
63+
loader: 'custom',
64+
loaderFile: './custom-loader.js',
65+
},
66+
}
67+
68+
await setImageConfig(ctx.pluginContext)
69+
70+
expect(ctx.pluginContext.netlifyConfig.redirects).not.toEqual(
71+
expect.arrayContaining([
72+
{
73+
from: '/_next/image',
74+
query: {
75+
q: ':quality',
76+
url: ':url',
77+
w: ':width',
78+
},
79+
to: '/.netlify/images?url=:url&w=:width&q=:quality',
80+
status: 200,
81+
},
82+
]),
83+
)
84+
})
85+
86+
test<ImageCDNTestContext>('handles custom images.path', async (ctx) => {
87+
ctx.mockNextConfig = {
88+
images: {
89+
// Next.js automatically adds basePath to images.path (when user does not set custom `images.path` in their config)
90+
// if user sets custom `images.path` - it will be used as-is (so user need to cover their basePath by themselves
91+
// if they want to have it in their custom image endpoint
92+
// see https://github.com/vercel/next.js/blob/bb105ef4fbfed9d96a93794eeaed956eda2116d8/packages/next/src/server/config.ts#L426-L432)
93+
// either way `images.path` we get is final config with everything combined so we want to use it as-is
94+
path: '/base/path/_custom/image/endpoint',
95+
loader: 'default',
96+
},
97+
}
98+
99+
await setImageConfig(ctx.pluginContext)
100+
101+
expect(ctx.pluginContext.netlifyConfig.redirects).toEqual(
102+
expect.arrayContaining([
103+
{
104+
from: '/base/path/_custom/image/endpoint',
105+
query: {
106+
q: ':quality',
107+
url: ':url',
108+
w: ':width',
109+
},
110+
to: '/.netlify/images?url=:url&w=:width&q=:quality',
111+
status: 200,
112+
},
113+
]),
114+
)
115+
})
116+
})
117+
/* eslint-enable id-length */

src/build/image-cdn.ts

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
import { PluginContext } from './plugin-context.js'
2+
3+
/**
4+
* Rewrite next/image to netlify image cdn
5+
*/
6+
export const setImageConfig = async (ctx: PluginContext): Promise<void> => {
7+
const {
8+
images: { path: imageEndpointPath, loader: imageLoader },
9+
} = await ctx.getBuildConfig()
10+
11+
if (imageLoader === 'default') {
12+
ctx.netlifyConfig.redirects.push({
13+
from: imageEndpointPath,
14+
// w and q are too short to be used as params with id-length rule
15+
// but we are forced to do so because of the next/image loader decides on their names
16+
// eslint-disable-next-line id-length
17+
query: { url: ':url', w: ':width', q: ':quality' },
18+
to: '/.netlify/images?url=:url&w=:width&q=:quality',
19+
status: 200,
20+
})
21+
}
22+
}

src/build/plugin-context.ts

+16-3
Original file line numberDiff line numberDiff line change
@@ -6,9 +6,14 @@ import { mkdir, readFile, writeFile } from 'node:fs/promises'
66
import { dirname, join, resolve } from 'node:path'
77
import { fileURLToPath } from 'node:url'
88

9-
import { NetlifyPluginConstants, NetlifyPluginOptions, NetlifyPluginUtils } from '@netlify/build'
10-
import { PrerenderManifest, RoutesManifest } from 'next/dist/build/index.js'
11-
import { MiddlewareManifest } from 'next/dist/build/webpack/plugins/middleware-plugin.js'
9+
import type {
10+
NetlifyPluginConstants,
11+
NetlifyPluginOptions,
12+
NetlifyPluginUtils,
13+
} from '@netlify/build'
14+
import type { PrerenderManifest, RoutesManifest } from 'next/dist/build/index.js'
15+
import type { MiddlewareManifest } from 'next/dist/build/webpack/plugins/middleware-plugin.js'
16+
import type { NextConfigComplete } from 'next/dist/server/config-shared.js'
1217

1318
const MODULE_DIR = fileURLToPath(new URL('.', import.meta.url))
1419
const PLUGIN_DIR = join(MODULE_DIR, '../..')
@@ -49,6 +54,7 @@ export type CacheEntry = {
4954

5055
export class PluginContext {
5156
utils: NetlifyPluginUtils
57+
netlifyConfig: NetlifyPluginOptions['netlifyConfig']
5258
pluginName: string
5359
pluginVersion: string
5460

@@ -114,6 +120,7 @@ export class PluginContext {
114120
this.pluginVersion = this.packageJSON.version
115121
this.constants = options.constants
116122
this.utils = options.utils
123+
this.netlifyConfig = options.netlifyConfig
117124
}
118125

119126
/** Resolves a path correctly with mono repository awareness */
@@ -135,6 +142,12 @@ export class PluginContext {
135142
)
136143
}
137144

145+
/** Get Next Config from build output **/
146+
async getBuildConfig(): Promise<NextConfigComplete> {
147+
return JSON.parse(await readFile(join(this.publishDir, 'required-server-files.json'), 'utf-8'))
148+
.config
149+
}
150+
138151
/**
139152
* Get Next.js routes manifest from the build output
140153
*/

src/index.ts

+2
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ import {
1212
} from './build/content/static.js'
1313
import { createEdgeHandlers } from './build/functions/edge.js'
1414
import { createServerHandler } from './build/functions/server.js'
15+
import { setImageConfig } from './build/image-cdn.js'
1516
import { PluginContext } from './build/plugin-context.js'
1617

1718
export const onPreBuild = async (options: NetlifyPluginOptions) => {
@@ -25,6 +26,7 @@ export const onBuild = async (options: NetlifyPluginOptions) => {
2526
if (!existsSync(ctx.publishDir)) {
2627
ctx.failBuild('Publish directory not found, please check your netlify.toml')
2728
}
29+
await setImageConfig(ctx)
2830
await saveBuildCache(ctx)
2931

3032
await Promise.all([

tests/e2e/simple-app.test.ts

+14
Original file line numberDiff line numberDiff line change
@@ -40,3 +40,17 @@ test('Redirects correctly', async ({ page }) => {
4040
await page.goto(`${ctx.url}/redirect`)
4141
await expect(page).toHaveURL(`https://www.netlify.com/`)
4242
})
43+
44+
test('next/image is using Netlify Image CDN', async ({ page }) => {
45+
const nextImageResponsePromise = page.waitForResponse('**/_next/image**')
46+
47+
await page.goto(`${ctx.url}/image`)
48+
49+
const nextImageResponse = await nextImageResponsePromise
50+
expect(nextImageResponse.request().url()).toContain('_next/image?url=%2Fsquirrel.jpg')
51+
// ensure next/image is using Image CDN
52+
// source image is jpg, but when requesting it through Image CDN avif will be returned
53+
await expect(await nextImageResponse.headerValue('content-type')).toEqual('image/avif')
54+
55+
await expectImageWasLoaded(page.locator('img'))
56+
})
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
import Image from 'next/image'
2+
3+
export default function NextImageUsingNetlifyImageCDN() {
4+
return (
5+
<main>
6+
<h1>Next/Image + Netlify Image CDN</h1>
7+
<Image src="/squirrel.jpg" alt="a cute squirrel (next/image)" width={300} height={278} />
8+
</main>
9+
)
10+
}

tests/integration/simple-app.test.ts

+1
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,7 @@ test<FixtureTestContext>('Test that the simple next app is working', async (ctx)
3939
'404',
4040
'404.html',
4141
'500.html',
42+
'image',
4243
'index',
4344
'other',
4445
'redirect',

tests/utils/fixture.ts

+3
Original file line numberDiff line numberDiff line change
@@ -147,6 +147,9 @@ export async function runPluginStep(
147147
// INTERNAL_FUNCTIONS_SRC: '.netlify/functions-internal',
148148
// INTERNAL_EDGE_FUNCTIONS_SRC: '.netlify/edge-functions',
149149
},
150+
netlifyConfig: {
151+
redirects: [],
152+
},
150153
utils: {
151154
build: {
152155
failBuild: (message, options) => {

0 commit comments

Comments
 (0)