Skip to content

Commit f7faa7c

Browse files
authoredJan 25, 2025··
fix(ts/fast-strip): More robust generic arrow handling (#9913)
**Related issue:** - Closes #9878
1 parent 04333aa commit f7faa7c

File tree

5 files changed

+127
-30
lines changed

5 files changed

+127
-30
lines changed
 

‎.changeset/grumpy-clocks-boil.md

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
---
2+
swc_fast_ts_strip: patch
3+
swc_core: patch
4+
---
5+
6+
fix(es/ts_strip): More robust generic arrow handling

‎crates/swc_fast_ts_strip/src/lib.rs

+69-28
Original file line numberDiff line numberDiff line change
@@ -15,11 +15,11 @@ use swc_ecma_ast::{
1515
Constructor, Decl, DefaultDecl, DoWhileStmt, EsVersion, ExportAll, ExportDecl,
1616
ExportDefaultDecl, ExportSpecifier, FnDecl, ForInStmt, ForOfStmt, ForStmt, GetterProp, IfStmt,
1717
ImportDecl, ImportSpecifier, NamedExport, ObjectPat, Param, Pat, PrivateMethod, PrivateProp,
18-
Program, ReturnStmt, SetterProp, Stmt, TsAsExpr, TsConstAssertion, TsEnumDecl,
18+
Program, ReturnStmt, SetterProp, Stmt, ThrowStmt, TsAsExpr, TsConstAssertion, TsEnumDecl,
1919
TsExportAssignment, TsImportEqualsDecl, TsIndexSignature, TsInstantiation, TsModuleDecl,
2020
TsModuleName, TsNamespaceDecl, TsNonNullExpr, TsParamPropParam, TsSatisfiesExpr,
2121
TsTypeAliasDecl, TsTypeAnn, TsTypeAssertion, TsTypeParamDecl, TsTypeParamInstantiation,
22-
VarDeclarator, WhileStmt,
22+
VarDeclarator, WhileStmt, YieldExpr,
2323
};
2424
use swc_ecma_parser::{
2525
lexer::Lexer,
@@ -596,6 +596,34 @@ impl TsStrip {
596596

597597
self.add_replacement(*span);
598598
}
599+
600+
// ```TypeScript
601+
// return/yield/throw <T>
602+
// (v: T) => v;
603+
// ```
604+
//
605+
// ```TypeScript
606+
// return/yield/throw (
607+
// v ) => v;
608+
// ```
609+
fn fix_asi_in_arrow_expr(&mut self, arrow_expr: &ArrowExpr) {
610+
if let Some(tp) = &arrow_expr.type_params {
611+
let l_paren = self.get_next_token(tp.span.hi);
612+
debug_assert_eq!(l_paren.token, Token::LParen);
613+
614+
let slice = self.get_src_slice(tp.span.with_hi(l_paren.span.lo));
615+
616+
if !slice.chars().any(is_new_line) {
617+
return;
618+
}
619+
620+
let l_paren_pos = l_paren.span.lo;
621+
let l_lt_pos = tp.span.lo;
622+
623+
self.add_overwrite(l_paren_pos, b' ');
624+
self.add_overwrite(l_lt_pos, b'(');
625+
}
626+
}
599627
}
600628

601629
impl Visit for TsStrip {
@@ -695,39 +723,52 @@ impl Visit for TsStrip {
695723

696724
arg.visit_with(self);
697725

698-
if let Some(arrow_expr) = arg.as_arrow() {
699-
if arrow_expr.is_async {
700-
// We have already handled type parameters in `visit_arrow_expr`.
701-
return;
702-
}
726+
let Some(arrow_expr) = arg.as_arrow() else {
727+
return;
728+
};
703729

704-
// ```TypeScript
705-
// return <T>
706-
// (v: T) => v;
707-
// ```
708-
//
709-
// ```TypeScript
710-
// return (
711-
// v ) => v;
712-
// ```
730+
if arrow_expr.is_async {
731+
// We have already handled type parameters in `visit_arrow_expr`.
732+
return;
733+
}
713734

714-
if let Some(tp) = &arrow_expr.type_params {
715-
let l_paren = self.get_next_token(tp.span.hi);
716-
debug_assert_eq!(l_paren.token, Token::LParen);
735+
self.fix_asi_in_arrow_expr(arrow_expr);
736+
}
717737

718-
let slice = self.get_src_slice(tp.span.with_hi(l_paren.span.lo));
738+
fn visit_yield_expr(&mut self, n: &YieldExpr) {
739+
let Some(arg) = &n.arg else {
740+
return;
741+
};
719742

720-
if !slice.chars().any(is_new_line) {
721-
return;
722-
}
743+
arg.visit_with(self);
723744

724-
let l_paren_pos = l_paren.span.lo;
725-
let l_lt_pos = tp.span.lo;
745+
let Some(arrow_expr) = arg.as_arrow() else {
746+
return;
747+
};
726748

727-
self.add_overwrite(l_paren_pos, b' ');
728-
self.add_overwrite(l_lt_pos, b'(');
729-
}
749+
if arrow_expr.is_async {
750+
// We have already handled type parameters in `visit_arrow_expr`.
751+
return;
730752
}
753+
754+
self.fix_asi_in_arrow_expr(arrow_expr);
755+
}
756+
757+
fn visit_throw_stmt(&mut self, n: &ThrowStmt) {
758+
let arg = &n.arg;
759+
760+
arg.visit_with(self);
761+
762+
let Some(arrow_expr) = arg.as_arrow() else {
763+
return;
764+
};
765+
766+
if arrow_expr.is_async {
767+
// We have already handled type parameters in `visit_arrow_expr`.
768+
return;
769+
}
770+
771+
self.fix_asi_in_arrow_expr(arrow_expr);
731772
}
732773

733774
fn visit_binding_ident(&mut self, n: &BindingIdent) {

‎crates/swc_fast_ts_strip/tests/fixture/issue-9878.js

+20-1
Original file line numberDiff line numberDiff line change
@@ -15,4 +15,23 @@ function f3() {
1515

1616
x
1717
)=>x;
18-
}
18+
}
19+
20+
(function () {
21+
return(
22+
v ) => v
23+
});
24+
(function () {
25+
return/**/(
26+
27+
/**/ v /**/
28+
)/**/=> v
29+
});
30+
(function* () {
31+
yield(
32+
v )=>v;
33+
});
34+
(function* () {
35+
throw(
36+
v )=>v;
37+
});

‎crates/swc_fast_ts_strip/tests/fixture/issue-9878.transform.js

+12
Original file line numberDiff line numberDiff line change
@@ -7,3 +7,15 @@ function f2() {
77
function f3() {
88
return (x)=>x;
99
}
10+
(function() {
11+
return (v)=>v;
12+
});
13+
(function() {
14+
return (v)=>v;
15+
});
16+
(function*() {
17+
yield (v)=>v;
18+
});
19+
(function*() {
20+
throw (v)=>v;
21+
});

‎crates/swc_fast_ts_strip/tests/fixture/issue-9878.ts

+20-1
Original file line numberDiff line numberDiff line change
@@ -15,4 +15,23 @@ function f3() {
1515
>
1616
(x: T): Promise<
1717
T>=>x;
18-
}
18+
}
19+
20+
(function () {
21+
return<T>
22+
(v: T) => v
23+
});
24+
(function () {
25+
return/**/<
26+
T
27+
>/**/(v: T)/**/:
28+
T/**/=> v
29+
});
30+
(function* () {
31+
yield<T>
32+
(v: T)=>v;
33+
});
34+
(function* () {
35+
throw<T>
36+
(v: T)=>v;
37+
});

0 commit comments

Comments
 (0)
Please sign in to comment.