7
7
createFilter ,
8
8
getNpmPackageName ,
9
9
isBuiltin ,
10
+ isInNodeModules ,
10
11
} from './utils'
11
12
import type { Environment } from './environment'
12
13
import type { PartialEnvironment } from './baseEnvironment'
@@ -15,14 +16,14 @@ const debug = createDebugger('vite:external')
15
16
16
17
const isExternalCache = new WeakMap <
17
18
Environment ,
18
- ( id : string , importer ?: string ) => boolean | undefined
19
+ ( id : string , importer ?: string ) => boolean
19
20
> ( )
20
21
21
22
export function shouldExternalize (
22
23
environment : Environment ,
23
24
id : string ,
24
25
importer : string | undefined ,
25
- ) : boolean | undefined {
26
+ ) : boolean {
26
27
let isExternal = isExternalCache . get ( environment )
27
28
if ( ! isExternal ) {
28
29
isExternal = createIsExternal ( environment )
@@ -72,27 +73,31 @@ export function createIsConfiguredAsExternal(
72
73
73
74
const isExternalizable = (
74
75
id : string ,
75
- importer ? : string ,
76
- configuredAsExternal ? : boolean ,
76
+ importer : string | undefined ,
77
+ configuredAsExternal : boolean ,
77
78
) : boolean => {
78
79
if ( ! bareImportRE . test ( id ) || id . includes ( '\0' ) ) {
79
80
return false
80
81
}
81
82
try {
82
- return ! ! tryNodeResolve (
83
+ const resolved = tryNodeResolve (
83
84
id ,
84
85
// Skip passing importer in build to avoid externalizing non-hoisted dependencies
85
86
// unresolvable from root (which would be unresolvable from output bundles also)
86
87
config . command === 'build' ? undefined : importer ,
87
88
resolveOptions ,
88
89
undefined ,
89
- // try to externalize, will return undefined or an object without
90
- // a external flag if it isn't externalizable
91
- true ,
92
- // Allow linked packages to be externalized if they are explicitly
93
- // configured as external
94
- ! ! configuredAsExternal ,
95
- ) ?. external
90
+ false ,
91
+ )
92
+ if ( ! resolved ) {
93
+ return false
94
+ }
95
+ // Only allow linked packages to be externalized
96
+ // if they are explicitly configured as external
97
+ if ( ! configuredAsExternal && ! isInNodeModules ( resolved . id ) ) {
98
+ return false
99
+ }
100
+ return canExternalizeFile ( resolved . id )
96
101
} catch {
97
102
debug ?.(
98
103
`Failed to node resolve "${ id } ". Skipping externalizing it by default.` ,
@@ -115,7 +120,7 @@ export function createIsConfiguredAsExternal(
115
120
}
116
121
const pkgName = getNpmPackageName ( id )
117
122
if ( ! pkgName ) {
118
- return isExternalizable ( id , importer )
123
+ return isExternalizable ( id , importer , false )
119
124
}
120
125
if (
121
126
// A package name in ssr.external externalizes every
@@ -139,14 +144,14 @@ export function createIsConfiguredAsExternal(
139
144
140
145
function createIsExternal (
141
146
environment : Environment ,
142
- ) : ( id : string , importer ?: string ) => boolean | undefined {
143
- const processedIds = new Map < string , boolean | undefined > ( )
147
+ ) : ( id : string , importer ?: string ) => boolean {
148
+ const processedIds = new Map < string , boolean > ( )
144
149
145
150
const isConfiguredAsExternal = createIsConfiguredAsExternal ( environment )
146
151
147
152
return ( id : string , importer ?: string ) => {
148
153
if ( processedIds . has ( id ) ) {
149
- return processedIds . get ( id )
154
+ return processedIds . get ( id ) !
150
155
}
151
156
let isExternal = false
152
157
if ( id [ 0 ] !== '.' && ! path . isAbsolute ( id ) ) {
@@ -156,3 +161,9 @@ function createIsExternal(
156
161
return isExternal
157
162
}
158
163
}
164
+
165
+ export function canExternalizeFile ( filePath : string ) : boolean {
166
+ const ext = path . extname ( filePath )
167
+ // only external js imports
168
+ return ! ext || ext === '.js' || ext === '.mjs' || ext === '.cjs'
169
+ }
0 commit comments