Skip to content

Commit 9cd4334

Browse files
authored
fix(es/compat): Handle sparse arrays correctly in generator transforms (#11131)
**Description:** Transforms arrays with empty slots correctly. ```javascript function* generator() {yield [, 'hello']} ``` is transformed to: ```javascript // ... function generator() { return _ts_generator(this, function(_state) { switch(_state.label){ case 0: return [ 4, [ 'hello' ] ]; case 1: _state.sent(); return [ 2 ]; } }); } ``` instead of: ```javascript function gen() { return _ts_generator(this, function (_state) { switch (_state.label) { case 0: return [ 4, [ , 'hello' ] ]; case 1: _state.sent(); return [ 2 ]; } }); } ``` (playground [url](https://play.swc.rs/?version=1.13.20&code=H4sIAAAAAAAAA0srzUsuyczP01JIT81LLUosyS%2FS0FSorsxMzUlRiNZRUM9IzcnJV4%2BtBQBOVRuJKQAAAA%3D%3D&config=H4sIAAAAAAAAA1WPSw7DIAwF9zkF8rrbdtE79BAWdSIifrKJVBTl7iUE0maH3xsz8jooBbNoeKq1PMsQkYX4nEsi2Sf8lARIOxTNJia49XaWvRrRCtVoOxpIyBOluiX3hoMNQajjLXPGmzH%2FC3VwkUnkCu4o%2BsnSVTc0JbjwXmrZDkk50qF%2FwA%2FqsvNjMPLqm4kXGrYvhlQioBQBAAA%3D) with 1.13.20)
1 parent 1c9ab27 commit 9cd4334

File tree

5 files changed

+126
-8
lines changed

5 files changed

+126
-8
lines changed

.changeset/shiny-spiders-relax.md

Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
---
2+
swc_ecma_compat_es2015: patch
3+
swc_core: patch
4+
---
5+
6+
fix(es/compat): Handle sparse arrays correctly in generator transforms

crates/swc_ecma_compat_es2015/src/generator.rs

Lines changed: 38 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -1112,7 +1112,7 @@ impl Generator {
11121112
);
11131113
}
11141114

1115-
let expressions = elements
1115+
let mut expressions = elements
11161116
.iter_mut()
11171117
.skip(num_initial_elements)
11181118
.map(|v| v.take())
@@ -1128,7 +1128,17 @@ impl Generator {
11281128
spread: None,
11291129
expr: Box::new(Expr::Array(ArrayLit {
11301130
span: DUMMY_SP,
1131-
elems: expressions,
1131+
elems: expressions
1132+
.take()
1133+
.into_iter()
1134+
.map(|expr| match expr {
1135+
Some(expr_or_spread) => match &*expr_or_spread.expr {
1136+
Expr::Invalid(_) => None,
1137+
_ => Some(expr_or_spread),
1138+
},
1139+
None => None,
1140+
})
1141+
.collect(),
11321142
})),
11331143
}],
11341144
..Default::default()
@@ -1141,7 +1151,13 @@ impl Generator {
11411151
.take()
11421152
.into_iter()
11431153
.map(Some)
1144-
.chain(expressions)
1154+
.chain(expressions.take().into_iter().map(|expr| match expr {
1155+
Some(expr_or_spread) => match &*expr_or_spread.expr {
1156+
Expr::Invalid(_) => None,
1157+
_ => Some(expr_or_spread),
1158+
},
1159+
None => None,
1160+
}))
11451161
.collect(),
11461162
}
11471163
.into()
@@ -1173,7 +1189,17 @@ impl Generator {
11731189
.as_callee(),
11741190
args: vec![Box::new(Expr::Array(ArrayLit {
11751191
span: DUMMY_SP,
1176-
elems: expressions.take(),
1192+
elems: expressions
1193+
.take()
1194+
.into_iter()
1195+
.map(|expr| match expr {
1196+
Some(expr_or_spread) => match &*expr_or_spread.expr {
1197+
Expr::Invalid(_) => None,
1198+
_ => Some(expr_or_spread),
1199+
},
1200+
None => None,
1201+
})
1202+
.collect(),
11771203
}))
11781204
.as_arg()],
11791205
..Default::default()
@@ -1187,7 +1213,13 @@ impl Generator {
11871213
.take()
11881214
.into_iter()
11891215
.map(Some)
1190-
.chain(expressions.take())
1216+
.chain(expressions.take().into_iter().map(|expr| match expr {
1217+
Some(expr_or_spread) => match &*expr_or_spread.expr {
1218+
Expr::Invalid(_) => None,
1219+
_ => Some(expr_or_spread),
1220+
},
1221+
None => None,
1222+
}))
11911223
.collect(),
11921224
}
11931225
.into(),
@@ -1199,9 +1231,7 @@ impl Generator {
11991231
}
12001232

12011233
element.visit_mut_with(self);
1202-
if element.is_some() {
1203-
expressions.push(element);
1204-
}
1234+
expressions.push(element);
12051235
expressions
12061236
}
12071237

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
function gen() {
2+
return _ts_generator(this, function (_state) {
3+
switch (_state.label) {
4+
case 0:
5+
return [
6+
4,
7+
[
8+
,
9+
'hello'
10+
]
11+
];
12+
case 1:
13+
_state.sent();
14+
return [
15+
2
16+
];
17+
}
18+
});
19+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
function gen() {
2+
return _ts_generator(this, function (_state) {
3+
switch (_state.label) {
4+
case 0:
5+
return [
6+
4,
7+
[
8+
'start',
9+
,
10+
'end'
11+
]
12+
];
13+
case 1:
14+
_state.sent();
15+
return [
16+
2
17+
];
18+
}
19+
});
20+
}
Lines changed: 43 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
#![allow(clippy::unit_arg)]
2+
3+
use swc_common::{comments::NoopComments, Mark};
4+
use swc_ecma_ast::Pass;
5+
use swc_ecma_parser::Syntax;
6+
use swc_ecma_transforms_base::resolver;
7+
use swc_ecma_transforms_compat::es2015::generator::generator;
8+
use swc_ecma_transforms_testing::test;
9+
10+
fn syntax() -> Syntax {
11+
Syntax::default()
12+
}
13+
14+
fn tr(_: ()) -> impl Pass {
15+
let unresolved_mark = Mark::new();
16+
let top_level_mark = Mark::new();
17+
(
18+
resolver(unresolved_mark, top_level_mark, false),
19+
generator(unresolved_mark, NoopComments),
20+
)
21+
}
22+
23+
test!(
24+
syntax(),
25+
|_| tr(Default::default()),
26+
sparse_array_leading_hole,
27+
r#"
28+
function* gen() {
29+
yield [, 'hello'];
30+
}
31+
"#
32+
);
33+
34+
test!(
35+
syntax(),
36+
|_| tr(Default::default()),
37+
sparse_array_middle_hole,
38+
r#"
39+
function* gen() {
40+
yield ['start', , 'end'];
41+
}
42+
"#
43+
);

0 commit comments

Comments
 (0)