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

feat: add the builtins environment resolve #18584

Merged
merged 21 commits into from
Jan 23, 2025
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
21 commits
Select commit Hold shift + click to select a range
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
12 changes: 9 additions & 3 deletions packages/vite/src/node/config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -61,16 +61,17 @@ import {
asyncFlatten,
createDebugger,
createFilter,
isBuiltin,
isExternalUrl,
isFilePathESM,
isInNodeModules,
isNodeBuiltin,
isNodeLikeBuiltin,
isObject,
isParentDirectory,
mergeAlias,
mergeConfig,
mergeWithDefaults,
nodeLikeBuiltins,
normalizeAlias,
normalizePath,
} from './utils'
Expand Down Expand Up @@ -881,7 +882,11 @@ function resolveEnvironmentResolveOptions(
consumer === 'client' || isSsrTargetWebworkerEnvironment
? DEFAULT_CLIENT_CONDITIONS
: DEFAULT_SERVER_CONDITIONS.filter((c) => c !== 'browser'),
enableBuiltinNoExternalCheck: !!isSsrTargetWebworkerEnvironment,
builtins:
resolve?.builtins ??
(consumer === 'server' && !isSsrTargetWebworkerEnvironment
? nodeLikeBuiltins
: []),
},
resolve ?? {},
)
Expand Down Expand Up @@ -1753,6 +1758,7 @@ async function bundleConfigFile(
preserveSymlinks: false,
packageCache,
isRequire,
builtins: nodeLikeBuiltins,
})?.id
}

Expand All @@ -1771,7 +1777,7 @@ async function bundleConfigFile(
// With the `isNodeBuiltin` check above, this check captures if the builtin is a
// non-node built-in, which esbuild doesn't know how to handle. In that case, we
// externalize it so the non-node runtime handles it instead.
if (isBuiltin(id)) {
if (isNodeLikeBuiltin(id)) {
return { external: true }
}

Expand Down
4 changes: 3 additions & 1 deletion packages/vite/src/node/external.ts
Original file line number Diff line number Diff line change
Expand Up @@ -155,7 +155,9 @@ function createIsExternal(
}
let isExternal = false
if (id[0] !== '.' && !path.isAbsolute(id)) {
isExternal = isBuiltin(id) || isConfiguredAsExternal(id, importer)
isExternal =
isBuiltin(environment.config.resolve.builtins, id) ||
isConfiguredAsExternal(id, importer)
}
processedIds.set(id, isExternal)
return isExternal
Expand Down
2 changes: 1 addition & 1 deletion packages/vite/src/node/optimizer/esbuildDepPlugin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -115,7 +115,7 @@ export function esbuildDepPlugin(
namespace: 'optional-peer-dep',
}
}
if (environment.config.consumer === 'server' && isBuiltin(resolved)) {
if (isBuiltin(environment.config.resolve.builtins, resolved)) {
return
}
Copy link
Member

Choose a reason for hiding this comment

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

I guess this does not work. cloudflare:* will be processed by esbuild and IIRC esbuild does not externalize them automatically and then it throws Could not resolve "cloudflare:*" error. I guess it needs to be return { path: resolved, external: true }.
But I wonder if we should set external: true for anything that was externalized by rollup plugins or resolve.external instead of checking isBuiltin here.

if (isExternalUrl(resolved)) {
Expand Down
2 changes: 1 addition & 1 deletion packages/vite/src/node/plugins/importAnalysis.ts
Original file line number Diff line number Diff line change
Expand Up @@ -519,7 +519,7 @@ export function importAnalysisPlugin(config: ResolvedConfig): Plugin {
if (shouldExternalize(environment, specifier, importer)) {
return
}
if (isBuiltin(specifier)) {
if (isBuiltin(environment.config.resolve.builtins, specifier)) {
return
}
}
Expand Down
117 changes: 70 additions & 47 deletions packages/vite/src/node/plugins/resolve.ts
Original file line number Diff line number Diff line change
Expand Up @@ -25,10 +25,12 @@ import {
isDataUrl,
isExternalUrl,
isInNodeModules,
isNodeLikeBuiltin,
isNonDriveRelativeAbsolutePath,
isObject,
isOptimizable,
isTsRequest,
nodeLikeBuiltins,
normalizePath,
safeRealpathSync,
tryStatSync,
Expand Down Expand Up @@ -97,9 +99,9 @@ export interface EnvironmentResolveOptions {
*/
external?: string[] | true
/**
* @internal
* Array of strings or regular expressions that indicate what modules are builtin for the environment.
*/
enableBuiltinNoExternalCheck?: boolean
builtins?: (string | RegExp)[]
}

export interface ResolveOptions extends EnvironmentResolveOptions {
Expand Down Expand Up @@ -173,11 +175,8 @@ interface ResolvePluginOptions {
}

export interface InternalResolveOptions
extends Required<Omit<ResolveOptions, 'enableBuiltinNoExternalCheck'>>,
ResolvePluginOptions {
/** @internal this is always optional for backward compat */
enableBuiltinNoExternalCheck?: boolean
}
extends Required<ResolveOptions>,
ResolvePluginOptions {}

// Defined ResolveOptions are used to overwrite the values for all environments
// It is used when creating custom resolvers (for CSS, scanning, etc)
Expand Down Expand Up @@ -422,47 +421,68 @@ export function resolvePlugin(
return res
}

// node built-ins.
// externalize if building for a node compatible environment, otherwise redirect to empty module
if (isBuiltin(id)) {
if (currentEnvironmentOptions.consumer === 'server') {
if (
options.enableBuiltinNoExternalCheck &&
options.noExternal === true &&
// if both noExternal and external are true, noExternal will take the higher priority and bundle it.
// only if the id is explicitly listed in external, we will externalize it and skip this error.
(options.external === true || !options.external.includes(id))
) {
let message = `Cannot bundle Node.js built-in "${id}"`
if (importer) {
message += ` imported from "${path.relative(
process.cwd(),
importer,
)}"`
}
message += `. Consider disabling environments.${this.environment.name}.noExternal or remove the built-in dependency.`
this.error(message)
// built-ins
// externalize if building for a server environment, otherwise redirect to an empty module
if (
currentEnvironmentOptions.consumer === 'server' &&
isBuiltin(options.builtins, id)
) {
return options.idOnly
? id
: { id, external: true, moduleSideEffects: false }
} else if (
currentEnvironmentOptions.consumer === 'server' &&
isNodeLikeBuiltin(id)
) {
if (
options.noExternal === true &&
// if both noExternal and external are true, noExternal will take the higher priority and bundle it.
// only if the id is explicitly listed in external, we will externalize it and skip this error.
(options.external === true || !options.external.includes(id))
) {
let message = `Cannot bundle node built-in module "${id}"`
if (importer) {
message += ` imported from "${path.relative(
process.cwd(),
importer,
)}"`
}

return options.idOnly
? id
: { id, external: true, moduleSideEffects: false }
} else {
if (!asSrc) {
debug?.(
`externalized node built-in "${id}" to empty module. ` +
`(imported by: ${colors.white(colors.dim(importer))})`,
)
} else if (isProduction) {
this.warn(
`Module "${id}" has been externalized for browser compatibility, imported by "${importer}". ` +
`See https://vite.dev/guide/troubleshooting.html#module-externalized-for-browser-compatibility for more details.`,
)
message += `. Consider disabling environments.${this.environment.name}.noExternal or remove the built-in dependency.`
this.error(message)
}
} else if (
currentEnvironmentOptions.consumer === 'client' &&
isNodeLikeBuiltin(id)
) {
Copy link
Member

@sapphi-red sapphi-red Dec 3, 2024

Choose a reason for hiding this comment

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

Suggested change
} else if (
currentEnvironmentOptions.consumer === 'client' &&
isNodeLikeBuiltin(id)
) {
} else if (
currentEnvironmentOptions.consumer === 'server' &&
isNodeLikeBuiltin(id)
) {
if (
options.noExternal === true &&
// if both noExternal and external are true, noExternal will take the higher priority and bundle it.
// only if the id is explicitly listed in external, we will externalize it and skip this error.
(options.external === true || !options.external.includes(id))
) {
let message = `Cannot bundle node built-in module "${id}"`
if (importer) {
message += ` imported from "${path.relative(
process.cwd(),
importer,
)}"`
}
message += `. Consider disabling environments.${this.environment.name}.noExternal or remove the built-in dependency.`
this.error(message)
}
} if (
currentEnvironmentOptions.consumer === 'client' &&
isNodeLikeBuiltin(id)
) {

How about moving the "if options.enableBuiltinNoExternalCheck block" here from above and removing options.enableBuiltinNoExternalCheck && and setting builtin: [] by default for ssr.target: 'webworker'?

By setting builtin: [], I mean updating this line to (consumer === 'server' && !isSsrTargetWebworkerEnvironment.
https://github.com/vitejs/vite/pull/18584/files#diff-11e17761d4ecfee8f8fde15c6d79b7bc0260176396a30dfd8e6f6bbaf5af4745R888

This way if an environment sets builtin and have some node builtin modules not included in builtin, Vite will show a more friendly error (Cannot bundle built-in module) when noExternal is set.

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've made the changes in regards to enableBuiltinNoExternalCheck that you suggested, it sounds sensible to me 🙂👍

if (
options.noExternal === true &&
// if both noExternal and external are true, noExternal will take the higher priority and bundle it.
// only if the id is explicitly listed in external, we will externalize it and skip this error.
(options.external === true || !options.external.includes(id))
) {
let message = `Cannot bundle built-in module "${id}"`
if (importer) {
message += ` imported from "${path.relative(
process.cwd(),
importer,
)}"`
}
return isProduction
? browserExternalId
: `${browserExternalId}:${id}`
message += `. Consider disabling environments.${this.environment.name}.noExternal or remove the built-in dependency.`
this.error(message)
}

if (!asSrc) {
debug?.(
`externalized node built-in "${id}" to empty module. ` +
`(imported by: ${colors.white(colors.dim(importer))})`,
)
} else if (isProduction) {
this.warn(
`Module "${id}" has been externalized for browser compatibility, imported by "${importer}". ` +
`See https://vite.dev/guide/troubleshooting.html#module-externalized-for-browser-compatibility for more details.`,
)
}
return isProduction ? browserExternalId : `${browserExternalId}:${id}`
}
}

Expand Down Expand Up @@ -720,8 +740,11 @@ export function tryNodeResolve(
basedir = root
}

const isModuleBuiltin = (id: string) =>
isBuiltin(options?.builtins ?? nodeLikeBuiltins, id)

let selfPkg = null
if (!isBuiltin(id) && !id.includes('\0') && bareImportRE.test(id)) {
if (!isModuleBuiltin(id) && !id.includes('\0') && bareImportRE.test(id)) {
// check if it's a self reference dep.
const selfPackageData = findNearestPackageData(basedir, packageCache)
selfPkg =
Expand All @@ -738,7 +761,7 @@ export function tryNodeResolve(
// if so, we can resolve to a special id that errors only when imported.
if (
basedir !== root && // root has no peer dep
!isBuiltin(id) &&
!isModuleBuiltin(id) &&
!id.includes('\0') &&
bareImportRE.test(id)
) {
Expand Down
7 changes: 5 additions & 2 deletions packages/vite/src/node/ssr/fetchModule.ts
Original file line number Diff line number Diff line change
Expand Up @@ -28,8 +28,10 @@ export async function fetchModule(
importer?: string,
options: FetchModuleOptions = {},
): Promise<FetchResult> {
// builtins should always be externalized
if (url.startsWith('data:') || isBuiltin(url)) {
if (
url.startsWith('data:') ||
isBuiltin(environment.config.resolve.builtins, url)
) {
return { externalize: url, type: 'builtin' }
}

Expand Down Expand Up @@ -57,6 +59,7 @@ export async function fetchModule(
isProduction,
root,
packageCache: environment.config.packageCache,
builtins: environment.config.resolve.builtins,
})
if (!resolved) {
const err: any = new Error(
Expand Down
42 changes: 37 additions & 5 deletions packages/vite/src/node/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -102,11 +102,43 @@ const BUN_BUILTIN_NAMESPACE = 'bun:'
// Some runtimes like Bun injects namespaced modules here, which is not a node builtin
const nodeBuiltins = builtinModules.filter((id) => !id.includes(':'))

// TODO: Use `isBuiltin` from `node:module`, but Deno doesn't support it
export function isBuiltin(id: string): boolean {
if (process.versions.deno && id.startsWith(NPM_BUILTIN_NAMESPACE)) return true
if (process.versions.bun && id.startsWith(BUN_BUILTIN_NAMESPACE)) return true
return isNodeBuiltin(id)
const isBuiltinCache = new WeakMap<
(string | RegExp)[],
(id: string, importer?: string) => boolean
>()

export function isBuiltin(builtins: (string | RegExp)[], id: string): boolean {
let isBuiltin = isBuiltinCache.get(builtins)
if (!isBuiltin) {
isBuiltin = createIsBuiltin(builtins)
isBuiltinCache.set(builtins, isBuiltin)
}
return isBuiltin(id)
}

export function createIsBuiltin(
builtins: (string | RegExp)[],
): (id: string) => boolean {
const plainBuiltinsSet = new Set(
builtins.filter((builtin) => typeof builtin === 'string'),
)
const regexBuiltins = builtins.filter(
(builtin) => typeof builtin !== 'string',
)

return (id) =>
plainBuiltinsSet.has(id) || regexBuiltins.some((regexp) => regexp.test(id))
}

export const nodeLikeBuiltins = [
...nodeBuiltins,
new RegExp(`^${NODE_BUILTIN_NAMESPACE}`),
new RegExp(`^${NPM_BUILTIN_NAMESPACE}`),
new RegExp(`^${BUN_BUILTIN_NAMESPACE}`),
]

export function isNodeLikeBuiltin(id: string): boolean {
return isBuiltin(nodeLikeBuiltins, id)
}

export function isNodeBuiltin(id: string): boolean {
Expand Down
Loading