Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

perf: memoize blobs requests in the request scope #2777

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions .eslintrc.cjs
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,14 @@ module.exports = {
message: 'Please use `getTracer()` from `./handlers/tracer.cjs` instead',
},
],
patterns: [
{
// only */storage/storage.cjs is allowed to be imported
// rest are implementation details that should not be used directly
group: ['*/storage/*', '!*/storage/storage.cjs'],
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should we use storage/index.cjs?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I personally prefer not to because my personal experience is that using index modules makes navigation around code base harder when you start to get multiple modules with same name in different directories, as well as just seeing multiple index modules opened in IDE tabs hellish so I do try to avoid it

similar example (just instead of index module, is Next's app router pattern to have page.j/ts(x)):
image

I can appreciate that storage/storage.cjs looks pretty weird, but I think it's better than added mental load of tracking multiple index modules. Additionally with IDEs auto-adding imports (like https://code.visualstudio.com/docs/languages/javascript#_auto-imports ) it's a bit "out of mind" as nowadays you usually don't even write your import statements manually

Some compromise could be to have index that just re-exports and doesn't actually have any logic inside, but then it just pollute directory tree IMO

message: 'Import public `[...]/storage/storage.cjs` module instead.',
},
],
},
],
},
Expand Down
76 changes: 30 additions & 46 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@
"fs-monkey": "^1.0.6",
"get-port": "^7.1.0",
"lambda-local": "^2.2.0",
"lru-cache": "^10.4.3",
"memfs": "^4.9.2",
"mock-require": "^3.0.3",
"msw": "^2.0.7",
Expand Down
2 changes: 1 addition & 1 deletion src/build/content/static.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import { trace } from '@opentelemetry/api'
import { wrapTracer } from '@opentelemetry/api/experimental'
import glob from 'fast-glob'

import type { HtmlBlob } from '../../run/next.cjs'
import type { HtmlBlob } from '../../shared/blob-types.cjs'
import { encodeBlobKey } from '../../shared/blobkey.js'
import { PluginContext } from '../plugin-context.js'
import { verifyNetlifyForms } from '../verification.js'
Expand Down
32 changes: 26 additions & 6 deletions src/run/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import { join, resolve } from 'node:path'
import type { NextConfigComplete } from 'next/dist/server/config-shared.js'

import { PLUGIN_DIR, RUN_CONFIG } from './constants.js'
import { setInMemoryCacheMaxSizeFromNextConfig } from './storage/storage.cjs'

/**
* Get Next.js config from the build output
Expand All @@ -13,10 +14,27 @@ export const getRunConfig = async () => {
return JSON.parse(await readFile(resolve(PLUGIN_DIR, RUN_CONFIG), 'utf-8'))
}

type NextConfigForMultipleVersions = NextConfigComplete & {
experimental: NextConfigComplete['experimental'] & {
// those are pre 14.1.0 options that were moved out of experimental in // https://github.com/vercel/next.js/pull/57953/files#diff-c49c4767e6ed8627e6e1b8f96b141ee13246153f5e9142e1da03450c8e81e96fL311

// https://github.com/vercel/next.js/blob/v14.0.4/packages/next/src/server/config-shared.ts#L182-L183
// custom path to a cache handler to use
incrementalCacheHandlerPath?: string
// https://github.com/vercel/next.js/blob/v14.0.4/packages/next/src/server/config-shared.ts#L207-L212
/**
* In-memory cache size in bytes.
*
* If `isrMemoryCacheSize: 0` disables in-memory caching.
*/
isrMemoryCacheSize?: number
}
}

/**
* Configure the custom cache handler at request time
*/
export const setRunConfig = (config: NextConfigComplete) => {
export const setRunConfig = (config: NextConfigForMultipleVersions) => {
const cacheHandler = join(PLUGIN_DIR, '.netlify/dist/run/handlers/cache.cjs')
if (!existsSync(cacheHandler)) {
throw new Error(`Cache handler not found at ${cacheHandler}`)
Expand All @@ -25,15 +43,17 @@ export const setRunConfig = (config: NextConfigComplete) => {
// set the path to the cache handler
config.experimental = {
...config.experimental,
// @ts-expect-error incrementalCacheHandlerPath was removed from config type
// but we still need to set it for older Next.js versions
// Before Next.js 14.1.0 path to the cache handler was in experimental section, see NextConfigForMultipleVersions type
incrementalCacheHandlerPath: cacheHandler,
}

// Next.js 14.1.0 moved the cache handler from experimental to stable
// https://github.com/vercel/next.js/pull/57953/files#diff-c49c4767e6ed8627e6e1b8f96b141ee13246153f5e9142e1da03450c8e81e96fL311
// Next.js 14.1.0 moved the cache handler from experimental to stable, see NextConfigForMultipleVersions type
config.cacheHandler = cacheHandler
config.cacheMaxMemorySize = 0

// honor the in-memory cache size from next.config (either one set by user or Next.js default)
setInMemoryCacheMaxSizeFromNextConfig(
config.cacheMaxMemorySize ?? config.experimental?.isrMemoryCacheSize,
)

// set config
process.env.__NEXT_PRIVATE_STANDALONE_CONFIG = JSON.stringify(config)
Expand Down
Loading
Loading