Skip to content

Commit 128ad8f

Browse files
fnlctrlbluwy
authored andcommitted
fix: use string manipulation instead of regex to inject esbuild helpers (#14094)
1 parent b1b816a commit 128ad8f

File tree

4 files changed

+30
-17
lines changed

4 files changed

+30
-17
lines changed

packages/vite/src/node/plugins/esbuild.ts

+23-16
Original file line numberDiff line numberDiff line change
@@ -28,10 +28,9 @@ import { searchForWorkspaceRoot } from '../server/searchRoot'
2828

2929
const debug = createDebugger('vite:esbuild')
3030

31-
const INJECT_HELPERS_IIFE_RE =
32-
/^(.*?)((?:const|var)\s+\S+\s*=\s*function\s*\([^)]*\)\s*\{\s*"use strict";)/s
33-
const INJECT_HELPERS_UMD_RE =
34-
/^(.*?)(\(function\([^)]*\)\s*\{.+?amd.+?function\([^)]*\)\s*\{\s*"use strict";)/s
31+
// IIFE content looks like `var MyLib = function() {`. Spaces are removed when minified
32+
const IIFE_BEGIN_RE =
33+
/(const|var)\s+\S+\s*=\s*function\(\)\s*\{.*"use strict";/s
3534

3635
const validExtensionRE = /\.\w+$/
3736
const jsxExtensionsRE = /\.(?:j|t)sx\b/
@@ -333,22 +332,30 @@ export const buildEsbuildPlugin = (config: ResolvedConfig): Plugin => {
333332
if (config.build.lib) {
334333
// #7188, esbuild adds helpers out of the UMD and IIFE wrappers, and the
335334
// names are minified potentially causing collision with other globals.
336-
// We use a regex to inject the helpers inside the wrappers.
335+
// We inject the helpers inside the wrappers.
336+
// e.g. turn:
337+
// <esbuild helpers> (function(){ /*actual content/* })()
338+
// into:
339+
// (function(){ <esbuild helpers> /*actual content/* })()
340+
// Not using regex because it's too hard to rule out performance issues like #8738 #8099 #10900 #14065
341+
// Instead, using plain string index manipulation (indexOf, slice) which is simple and performant
337342
// We don't need to create a MagicString here because both the helpers and
338343
// the headers don't modify the sourcemap
339-
const injectHelpers =
340-
opts.format === 'umd'
341-
? INJECT_HELPERS_UMD_RE
342-
: opts.format === 'iife'
343-
? INJECT_HELPERS_IIFE_RE
344-
: undefined
345-
if (injectHelpers) {
346-
res.code = res.code.replace(
347-
injectHelpers,
348-
(_, helpers, header) => header + helpers,
349-
)
344+
const esbuildCode = res.code
345+
const contentIndex =
346+
opts.format === 'iife'
347+
? esbuildCode.match(IIFE_BEGIN_RE)?.index || 0
348+
: opts.format === 'umd'
349+
? esbuildCode.indexOf(`(function(`) // same for minified or not
350+
: 0
351+
if (contentIndex > 0) {
352+
const esbuildHelpers = esbuildCode.slice(0, contentIndex)
353+
res.code = esbuildCode
354+
.slice(contentIndex)
355+
.replace(`"use strict";`, `"use strict";` + esbuildHelpers)
350356
}
351357
}
358+
352359
return res
353360
},
354361
}

playground/lib/__tests__/lib.spec.ts

+3-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,9 @@ describe.runIf(isBuild)('build', () => {
3333
'dist/nominify/my-lib-custom-filename.iife.js',
3434
)
3535
// esbuild helpers are injected inside of the IIFE wrapper
36-
expect(code).toMatch(/^var MyLib=function\(\)\{"use strict";/)
36+
// esbuild has a bug that injects some statements before `"use strict"`: https://github.com/evanw/esbuild/issues/3322
37+
// remove the `.*?` part once it's fixed
38+
expect(code).toMatch(/^var MyLib=function\(\)\{.*?"use strict";/)
3739
expect(noMinifyCode).toMatch(
3840
/^var MyLib\s*=\s*function\(\)\s*\{.*?"use strict";/s,
3941
)

playground/lib/src/main.js

+3
Original file line numberDiff line numberDiff line change
@@ -10,3 +10,6 @@ export default function myLib(sel) {
1010
// make sure umd helper has been moved to the right position
1111
console.log(`amd function(){ "use strict"; }`)
1212
}
13+
14+
// For triggering unhandled global esbuild helpers in previous regex-based implementation for injection
15+
;(function () {})()?.foo

playground/lib/vite.config.js

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ export default defineConfig({
77
supported: {
88
// Force esbuild inject helpers to test regex
99
'object-rest-spread': false,
10+
'optional-chain': false,
1011
},
1112
},
1213
build: {

0 commit comments

Comments
 (0)