Skip to content

Commit 55b9711

Browse files
authored
feat(legacy)!: bump modern target to support async generator (#11896)
1 parent d5b8f86 commit 55b9711

File tree

9 files changed

+105
-26
lines changed

9 files changed

+105
-26
lines changed

packages/plugin-legacy/README.md

+10-4
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ By default, this plugin will:
88

99
- Generate a polyfill chunk including SystemJS runtime, and any necessary polyfills determined by specified browser targets and **actual usage** in the bundle.
1010

11-
- Inject `<script nomodule>` tags into generated HTML to conditionally load the polyfills and legacy bundle only in browsers without native ESM support.
11+
- Inject `<script nomodule>` tags into generated HTML to conditionally load the polyfills and legacy bundle only in browsers without widely-available features support.
1212

1313
- Inject the `import.meta.env.LEGACY` env variable, which will only be `true` in the legacy production build, and `false` in all other cases.
1414

@@ -117,12 +117,18 @@ npm add -D terser
117117

118118
Defaults to `false`. Enabling this option will exclude `systemjs/dist/s.min.js` inside polyfills-legacy chunk.
119119

120-
## Dynamic Import
120+
## Browsers that supports ESM but does not support widely-available features
121121

122-
The legacy plugin offers a way to use native `import()` in the modern build while falling back to the legacy build in browsers with native ESM but without dynamic import support (e.g. Legacy Edge). This feature works by injecting a runtime check and loading the legacy bundle with SystemJs runtime if needed. There are the following drawbacks:
122+
The legacy plugin offers a way to use widely-available features natively in the modern build, while falling back to the legacy build in browsers with native ESM but without those features supported (e.g. Legacy Edge). This feature works by injecting a runtime check and loading the legacy bundle with SystemJs runtime if needed. There are the following drawbacks:
123123

124124
- Modern bundle is downloaded in all ESM browsers
125-
- Modern bundle throws `SyntaxError` in browsers without dynamic import
125+
- Modern bundle throws `SyntaxError` in browsers without those features support
126+
127+
The following syntax are considered as widely-available:
128+
129+
- dynamic import
130+
- `import.meta`
131+
- async generator
126132

127133
## Polyfill Specifiers
128134

packages/plugin-legacy/package.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -41,10 +41,10 @@
4141
},
4242
"homepage": "https://github.com/vitejs/vite/tree/main/packages/plugin-legacy#readme",
4343
"dependencies": {
44-
"core-js": "^3.27.2",
4544
"@babel/core": "^7.20.12",
4645
"@babel/preset-env": "^7.20.2",
4746
"browserslist": "^4.21.4",
47+
"core-js": "^3.27.2",
4848
"magic-string": "^0.27.0",
4949
"regenerator-runtime": "^0.13.11",
5050
"systemjs": "^6.13.0"
@@ -54,6 +54,7 @@
5454
"vite": "^4.0.0"
5555
},
5656
"devDependencies": {
57+
"acorn": "^8.8.2",
5758
"picocolors": "^1.0.0",
5859
"vite": "workspace:*"
5960
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
import { expect, test } from 'vitest'
2+
import type { ecmaVersion } from 'acorn'
3+
import { parse } from 'acorn'
4+
import { detectModernBrowserDetector } from '../snippets'
5+
6+
const shouldFailVersions: ecmaVersion[] = []
7+
for (let v = 2015; v <= 2019; v++) {
8+
shouldFailVersions.push(v as ecmaVersion)
9+
}
10+
11+
const shouldPassVersions: acorn.ecmaVersion[] = []
12+
for (let v = 2020; v <= 2022; v++) {
13+
shouldPassVersions.push(v as ecmaVersion)
14+
}
15+
16+
for (const version of shouldFailVersions) {
17+
test(`detect code should not be able to be parsed with ES${version}`, () => {
18+
expect(() => {
19+
parse(detectModernBrowserDetector, {
20+
ecmaVersion: version,
21+
sourceType: 'module',
22+
})
23+
}).toThrow()
24+
})
25+
}
26+
27+
for (const version of shouldPassVersions) {
28+
test(`detect code should be able to be parsed with ES${version}`, () => {
29+
expect(() => {
30+
parse(detectModernBrowserDetector, {
31+
ecmaVersion: version,
32+
sourceType: 'module',
33+
})
34+
}).not.toThrow()
35+
})
36+
}

packages/plugin-legacy/src/index.ts

+16-21
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,15 @@ import type {
2525
import colors from 'picocolors'
2626
import { loadConfig as browserslistLoadConfig } from 'browserslist'
2727
import type { Options } from './types'
28+
import {
29+
detectModernBrowserCode,
30+
dynamicFallbackInlineCode,
31+
legacyEntryId,
32+
legacyPolyfillId,
33+
modernChunkLegacyGuard,
34+
safari10NoModuleFix,
35+
systemJSInlineCode,
36+
} from './snippets'
2837

2938
// lazy load babel since it's not used during dev
3039
// eslint-disable-next-line @typescript-eslint/consistent-type-imports
@@ -103,20 +112,6 @@ function toAssetPathFromHtml(
103112
)
104113
}
105114

106-
// https://gist.github.com/samthor/64b114e4a4f539915a95b91ffd340acc
107-
// DO NOT ALTER THIS CONTENT
108-
const safari10NoModuleFix = `!function(){var e=document,t=e.createElement("script");if(!("noModule"in t)&&"onbeforeload"in t){var n=!1;e.addEventListener("beforeload",(function(e){if(e.target===t)n=!0;else if(!e.target.hasAttribute("nomodule")||!n)return;e.preventDefault()}),!0),t.type="module",t.src=".",e.head.appendChild(t),t.remove()}}();`
109-
110-
const legacyPolyfillId = 'vite-legacy-polyfill'
111-
const legacyEntryId = 'vite-legacy-entry'
112-
const systemJSInlineCode = `System.import(document.getElementById('${legacyEntryId}').getAttribute('data-src'))`
113-
114-
const detectModernBrowserVarName = '__vite_is_modern_browser'
115-
const detectModernBrowserCode = `try{import.meta.url;import("_").catch(()=>1);}catch(e){}window.${detectModernBrowserVarName}=true;`
116-
const dynamicFallbackInlineCode = `!function(){if(window.${detectModernBrowserVarName})return;console.warn("vite: loading legacy build because dynamic import or import.meta.url is unsupported, syntax error above should be ignored");var e=document.getElementById("${legacyPolyfillId}"),n=document.createElement("script");n.src=e.src,n.onload=function(){${systemJSInlineCode}},document.body.appendChild(n)}();`
117-
118-
const forceDynamicImportUsage = `export function __vite_legacy_guard(){import('data:text/javascript,')};`
119-
120115
const legacyEnvVarMarker = `__VITE_IS_LEGACY__`
121116

122117
const _require = createRequire(import.meta.url)
@@ -126,7 +121,6 @@ function viteLegacyPlugin(options: Options = {}): Plugin[] {
126121
let targets: Options['targets']
127122

128123
const genLegacy = options.renderLegacyChunks !== false
129-
const genDynamicFallback = genLegacy
130124

131125
const debugFlags = (process.env.DEBUG || '').split(',')
132126
const isDebug =
@@ -185,13 +179,13 @@ function viteLegacyPlugin(options: Options = {}): Plugin[] {
185179
// Vite's default target browsers are **not** the same.
186180
// See https://github.com/vitejs/vite/pull/10052#issuecomment-1242076461
187181
overriddenBuildTarget = config.build.target !== undefined
188-
// browsers supporting ESM + dynamic import + import.meta
182+
// browsers supporting ESM + dynamic import + import.meta + async generator
189183
config.build.target = [
190184
'es2020',
191185
'edge79',
192186
'firefox67',
193187
'chrome64',
194-
'safari11.1',
188+
'safari12',
195189
]
196190
}
197191
}
@@ -252,7 +246,7 @@ function viteLegacyPlugin(options: Options = {}): Plugin[] {
252246
}
253247

254248
// legacy bundle
255-
if (legacyPolyfills.size || genDynamicFallback) {
249+
if (legacyPolyfills.size) {
256250
// check if the target needs Promise polyfill because SystemJS relies on it
257251
// https://github.com/systemjs/systemjs#ie11-support
258252
await detectPolyfills(
@@ -367,8 +361,9 @@ function viteLegacyPlugin(options: Options = {}): Plugin[] {
367361

368362
const ms = new MagicString(raw)
369363

370-
if (genDynamicFallback && chunk.isEntry) {
371-
ms.prepend(forceDynamicImportUsage)
364+
if (genLegacy && chunk.isEntry) {
365+
// append this code to avoid modern chunks running on legacy targeted browsers
366+
ms.prepend(modernChunkLegacyGuard)
372367
}
373368

374369
if (raw.includes(legacyEnvVarMarker)) {
@@ -557,7 +552,7 @@ function viteLegacyPlugin(options: Options = {}): Plugin[] {
557552
}
558553

559554
// 5. inject dynamic import fallback entry
560-
if (genDynamicFallback && legacyPolyfillFilename && legacyEntryFilename) {
555+
if (genLegacy && legacyPolyfillFilename && legacyEntryFilename) {
561556
tags.push({
562557
tag: 'script',
563558
attrs: { type: 'module' },
+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
// https://gist.github.com/samthor/64b114e4a4f539915a95b91ffd340acc
2+
// DO NOT ALTER THIS CONTENT
3+
export const safari10NoModuleFix = `!function(){var e=document,t=e.createElement("script");if(!("noModule"in t)&&"onbeforeload"in t){var n=!1;e.addEventListener("beforeload",(function(e){if(e.target===t)n=!0;else if(!e.target.hasAttribute("nomodule")||!n)return;e.preventDefault()}),!0),t.type="module",t.src=".",e.head.appendChild(t),t.remove()}}();`
4+
5+
export const legacyPolyfillId = 'vite-legacy-polyfill'
6+
export const legacyEntryId = 'vite-legacy-entry'
7+
export const systemJSInlineCode = `System.import(document.getElementById('${legacyEntryId}').getAttribute('data-src'))`
8+
9+
const detectModernBrowserVarName = '__vite_is_modern_browser'
10+
export const detectModernBrowserDetector =
11+
'import.meta.url;import("_").catch(()=>1);async function* g(){};'
12+
export const detectModernBrowserCode = `${detectModernBrowserDetector}window.${detectModernBrowserVarName}=true;`
13+
export const dynamicFallbackInlineCode = `!function(){if(window.${detectModernBrowserVarName})return;console.warn("vite: loading legacy chunks, syntax error above and the same error below should be ignored");var e=document.getElementById("${legacyPolyfillId}"),n=document.createElement("script");n.src=e.src,n.onload=function(){${systemJSInlineCode}},document.body.appendChild(n)}();`
14+
15+
export const modernChunkLegacyGuard = `export function __vite_legacy_guard(){${detectModernBrowserDetector}};`

playground/legacy/__tests__/legacy.spec.ts

+8
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,14 @@ test('transpiles down iterators correctly', async () => {
3030
await untilUpdated(() => page.textContent('#iterators'), 'hello', true)
3131
})
3232

33+
test('async generator', async () => {
34+
await untilUpdated(
35+
() => page.textContent('#async-generator'),
36+
'[0,1,2]',
37+
true,
38+
)
39+
})
40+
3341
test('wraps with iife', async () => {
3442
await untilUpdated(
3543
() => page.textContent('#babel-helpers'),

playground/legacy/index.html

+1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ <h1 id="app"></h1>
22
<div id="env"></div>
33
<div id="iterators"></div>
44
<div id="features-after-corejs-3"></div>
5+
<div id="async-generator"></div>
56
<div id="babel-helpers"></div>
67
<div id="assets"></div>
78
<button id="dynamic-css-button">dynamic css</button>

playground/legacy/main.js

+15
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,21 @@ text(
2929
JSON.stringify(structuredClone({ foo: 'foo' })),
3030
)
3131

32+
// async generator
33+
async function* asyncGenerator() {
34+
for (let i = 0; i < 3; i++) {
35+
await new Promise((resolve) => setTimeout(resolve, 10))
36+
yield i
37+
}
38+
}
39+
;(async () => {
40+
const result = []
41+
for await (const i of asyncGenerator()) {
42+
result.push(i)
43+
}
44+
text('#async-generator', JSON.stringify(result))
45+
})()
46+
3247
// babel-helpers
3348
// Using `String.raw` to inject `@babel/plugin-transform-template-literals`
3449
// helpers.

pnpm-lock.yaml

+2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)