Skip to content

Commit 1c686e2

Browse files
feat: cache .next/cache between builds (#185)
* feat: cache .next/cache between builds * fix: `cwd` memoization * fix: fix Windows tests * fix: fix Windows tests Co-authored-by: ehmicky <[email protected]>
1 parent 663dea1 commit 1c686e2

File tree

6 files changed

+112
-5
lines changed

6 files changed

+112
-5
lines changed

helpers/cacheBuild.js

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
const path = require('path')
2+
3+
const DEFAULT_DIST_DIR = '.next'
4+
5+
// Account for possible custom distDir
6+
const getPath = (distDir, source) => {
7+
return path.join(distDir || DEFAULT_DIST_DIR, source)
8+
}
9+
10+
const restoreCache = async ({ cache, distDir }) => {
11+
const cacheDir = getPath(distDir, 'cache')
12+
if (await cache.restore(cacheDir)) {
13+
console.log('Next.js cache restored.')
14+
} else {
15+
console.log('No Next.js cache to restore.')
16+
}
17+
}
18+
19+
const saveCache = async ({ cache, distDir }) => {
20+
const cacheDir = getPath(distDir, 'cache')
21+
const buildManifest = getPath(distDir, 'build-manifest.json')
22+
if (await cache.save(cacheDir, { digests: [buildManifest] })) {
23+
console.log('Next.js cache saved.')
24+
} else {
25+
console.log('No Next.js cache to save.')
26+
}
27+
}
28+
29+
module.exports = {
30+
restoreCache,
31+
saveCache,
32+
}

helpers/getNextConfig.js

+12-4
Original file line numberDiff line numberDiff line change
@@ -1,24 +1,32 @@
11
'use strict'
22

3+
const { cwd: getCwd } = require('process')
34
const { resolve } = require('path')
45

56
const moize = require('moize')
67

7-
// Load next.config.js
8-
const getNextConfig = async function (failBuild = defaultFailBuild) {
8+
// We used to cache nextConfig for any cwd. Now we pass process.cwd() to cache
9+
// (or memoize) nextConfig per cwd.
10+
const getNextConfig = async function (failBuild = defaultFailBuild, cwd = getCwd()) {
911
// We cannot load `next` at the top-level because we validate whether the
1012
// site is using `next` inside `onPreBuild`.
1113
const { PHASE_PRODUCTION_BUILD } = require('next/constants')
1214
const loadConfig = require('next/dist/next-server/server/config').default
1315

1416
try {
15-
return await loadConfig(PHASE_PRODUCTION_BUILD, resolve('.'))
17+
return await loadConfig(PHASE_PRODUCTION_BUILD, cwd)
1618
} catch (error) {
1719
return failBuild('Error loading your next.config.js.', { error })
1820
}
1921
}
2022

21-
const moizedGetNextConfig = moize(getNextConfig, { maxSize: 1e3, isPromise: true })
23+
const moizedGetNextConfig = moize(getNextConfig, {
24+
maxSize: 1e3,
25+
isPromise: true,
26+
// Memoization cache key. We need to use `transformArgs` so `process.cwd()`
27+
// default value is assigned
28+
transformArgs: ([, cwd = getCwd()]) => [cwd],
29+
})
2230

2331
const defaultFailBuild = function (message, { error }) {
2432
throw new Error(`${message}\n${error.stack}`)

helpers/isStaticExportProject.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ const isStaticExportProject = ({ build, scripts }) => {
1717

1818
if (isStaticExport) {
1919
console.log(
20-
`Static HTML export Next.js projects do not require this plugin. Check your project's build command for 'next export'.`,
20+
'NOTE: Static HTML export Next.js projects (projects that use `next export`) do not require most of this plugin. For these sites, this plugin *only* caches builds.',
2121
)
2222
}
2323

index.js

+6
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ const validateNextUsage = require('./helpers/validateNextUsage')
99
const doesNotNeedPlugin = require('./helpers/doesNotNeedPlugin')
1010
const getNextConfig = require('./helpers/getNextConfig')
1111
const copyUnstableIncludedDirs = require('./helpers/copyUnstableIncludedDirs')
12+
const { restoreCache, saveCache } = require('./helpers/cacheBuild')
1213

1314
const pWriteFile = util.promisify(fs.writeFile)
1415

@@ -27,6 +28,9 @@ module.exports = {
2728
return failBuild('Could not find a package.json for this project')
2829
}
2930

31+
const nextConfig = await getNextConfig(utils.failBuild)
32+
await restoreCache({ cache: utils.cache, distDir: nextConfig.distDir })
33+
3034
if (await doesNotNeedPlugin({ netlifyConfig, packageJson, failBuild })) {
3135
return
3236
}
@@ -60,12 +64,14 @@ module.exports = {
6064

6165
await nextOnNetlify({ functionsDir: FUNCTIONS_SRC, publishDir: PUBLISH_DIR })
6266
},
67+
6368
async onPostBuild({ netlifyConfig, packageJson, constants: { FUNCTIONS_DIST }, utils }) {
6469
if (await doesNotNeedPlugin({ netlifyConfig, packageJson, utils })) {
6570
return
6671
}
6772

6873
const nextConfig = await getNextConfig(utils.failBuild)
74+
await saveCache({ cache: utils.cache, distDir: nextConfig.distDir })
6975
copyUnstableIncludedDirs({ nextConfig, functionsDist: FUNCTIONS_DIST })
7076
},
7177
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
module.exports = {
2+
target: 'serverless',
3+
distDir: 'build',
4+
}

test/index.js

+57
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,10 @@ const utils = {
1818
throw new Error(message)
1919
},
2020
},
21+
cache: {
22+
save() {},
23+
restore() {},
24+
},
2125
}
2226

2327
// Temporary switch cwd
@@ -155,6 +159,29 @@ describe('preBuild()', () => {
155159
}),
156160
).rejects.toThrow(`Error loading your next.config.js.`)
157161
})
162+
163+
test('restores cache with right paths', async () => {
164+
await useFixture('dist_dir_next_config')
165+
166+
let distPath
167+
const utils_ = {
168+
...utils,
169+
cache: {
170+
restore: (x) => (distPath = x),
171+
},
172+
}
173+
const spy = jest.spyOn(utils_.cache, 'restore')
174+
175+
await plugin.onPreBuild({
176+
netlifyConfig,
177+
packageJson: DUMMY_PACKAGE_JSON,
178+
utils: utils_,
179+
constants: { FUNCTIONS_SRC: 'out_functions' },
180+
})
181+
182+
expect(spy).toHaveBeenCalled()
183+
expect(path.normalize(distPath)).toBe(path.normalize('build/cache'))
184+
})
158185
})
159186

160187
describe('onBuild()', () => {
@@ -229,3 +256,33 @@ describe('onBuild()', () => {
229256
expect(await pathExists(`${resolvedFunctions}/next_api_test/next_api_test.js`)).toBeTruthy()
230257
})
231258
})
259+
260+
describe('onPostBuild', () => {
261+
test('saves cache with right paths', async () => {
262+
await useFixture('dist_dir_next_config')
263+
264+
let distPath
265+
let manifestPath
266+
const utils_ = {
267+
...utils,
268+
cache: {
269+
save: (x, y) => {
270+
distPath = x
271+
manifestPath = y
272+
},
273+
},
274+
}
275+
const spy = jest.spyOn(utils_.cache, 'save')
276+
277+
await plugin.onPostBuild({
278+
netlifyConfig,
279+
packageJson: DUMMY_PACKAGE_JSON,
280+
utils: utils_,
281+
constants: { FUNCTIONS_SRC: 'out_functions' },
282+
})
283+
284+
expect(spy).toHaveBeenCalled()
285+
expect(path.normalize(distPath)).toBe(path.normalize('build/cache'))
286+
expect(path.normalize(manifestPath.digests[0])).toBe(path.normalize('build/build-manifest.json'))
287+
})
288+
})

0 commit comments

Comments
 (0)