@@ -3,24 +3,65 @@ use oxc_ast::{AstKind, ast::*};
3
3
use oxc_semantic:: { AstNode , AstNodes , ReferenceId , Scoping , SymbolId } ;
4
4
use rustc_hash:: FxHashSet ;
5
5
6
- #[ cfg_attr( not( test) , expect( dead_code) ) ]
7
- pub fn collect_name_symbols ( scoping : & Scoping , ast_nodes : & AstNodes ) -> FxHashSet < SymbolId > {
8
- let collector = NameSymbolCollector :: new ( scoping, ast_nodes) ;
6
+ #[ derive( Debug , Clone , Copy , Default ) ]
7
+ pub struct MangleOptionsKeepNames {
8
+ /// Preserve `name` property for functions.
9
+ ///
10
+ /// Default `false`
11
+ pub function : bool ,
12
+
13
+ /// Preserve `name` property for classes.
14
+ ///
15
+ /// Default `false`
16
+ pub class : bool ,
17
+ }
18
+
19
+ impl MangleOptionsKeepNames {
20
+ pub fn all_false ( ) -> Self {
21
+ Self { function : false , class : false }
22
+ }
23
+
24
+ pub fn all_true ( ) -> Self {
25
+ Self { function : true , class : true }
26
+ }
27
+ }
28
+
29
+ impl From < bool > for MangleOptionsKeepNames {
30
+ fn from ( keep_names : bool ) -> Self {
31
+ if keep_names { Self :: all_true ( ) } else { Self :: all_false ( ) }
32
+ }
33
+ }
34
+
35
+ pub fn collect_name_symbols (
36
+ options : MangleOptionsKeepNames ,
37
+ scoping : & Scoping ,
38
+ ast_nodes : & AstNodes ,
39
+ ) -> FxHashSet < SymbolId > {
40
+ let collector = NameSymbolCollector :: new ( options, scoping, ast_nodes) ;
9
41
collector. collect ( )
10
42
}
11
43
12
44
/// Collects symbols that are used to set `name` properties of functions and classes.
13
45
struct NameSymbolCollector < ' a , ' b > {
46
+ options : MangleOptionsKeepNames ,
14
47
scoping : & ' b Scoping ,
15
48
ast_nodes : & ' b AstNodes < ' a > ,
16
49
}
17
50
18
51
impl < ' a , ' b : ' a > NameSymbolCollector < ' a , ' b > {
19
- fn new ( scoping : & ' b Scoping , ast_nodes : & ' b AstNodes < ' a > ) -> Self {
20
- Self { scoping, ast_nodes }
52
+ fn new (
53
+ options : MangleOptionsKeepNames ,
54
+ scoping : & ' b Scoping ,
55
+ ast_nodes : & ' b AstNodes < ' a > ,
56
+ ) -> Self {
57
+ Self { options, scoping, ast_nodes }
21
58
}
22
59
23
60
fn collect ( self ) -> FxHashSet < SymbolId > {
61
+ if !self . options . function && !self . options . class {
62
+ return FxHashSet :: default ( ) ;
63
+ }
64
+
24
65
self . scoping
25
66
. symbol_ids ( )
26
67
. filter ( |symbol_id| {
@@ -42,9 +83,12 @@ impl<'a, 'b: 'a> NameSymbolCollector<'a, 'b> {
42
83
fn is_name_set_declare_node ( & self , node : & ' a AstNode , symbol_id : SymbolId ) -> bool {
43
84
match node. kind ( ) {
44
85
AstKind :: Function ( function) => {
45
- function. id . as_ref ( ) . is_some_and ( |id| id. symbol_id ( ) == symbol_id)
86
+ self . options . function
87
+ && function. id . as_ref ( ) . is_some_and ( |id| id. symbol_id ( ) == symbol_id)
88
+ }
89
+ AstKind :: Class ( cls) => {
90
+ self . options . class && cls. id . as_ref ( ) . is_some_and ( |id| id. symbol_id ( ) == symbol_id)
46
91
}
47
- AstKind :: Class ( cls) => cls. id . as_ref ( ) . is_some_and ( |id| id. symbol_id ( ) == symbol_id) ,
48
92
AstKind :: VariableDeclarator ( decl) => {
49
93
if let BindingPatternKind :: BindingIdentifier ( id) = & decl. id . kind {
50
94
if id. symbol_id ( ) == symbol_id {
@@ -176,9 +220,18 @@ impl<'a, 'b: 'a> NameSymbolCollector<'a, 'b> {
176
220
}
177
221
}
178
222
179
- #[ expect( clippy:: unused_self) ]
180
223
fn is_expression_whose_name_needs_to_be_kept ( & self , expr : & Expression ) -> bool {
181
- expr. is_anonymous_function_definition ( )
224
+ let is_anonymous = expr. is_anonymous_function_definition ( ) ;
225
+ if !is_anonymous {
226
+ return false ;
227
+ }
228
+
229
+ if self . options . class && self . options . function {
230
+ return true ;
231
+ }
232
+
233
+ let is_class = matches ! ( expr, Expression :: ClassExpression ( _) ) ;
234
+ ( self . options . class && is_class) || ( self . options . function && !is_class)
182
235
}
183
236
}
184
237
@@ -189,81 +242,83 @@ mod test {
189
242
use oxc_semantic:: SemanticBuilder ;
190
243
use oxc_span:: SourceType ;
191
244
use rustc_hash:: FxHashSet ;
192
- use std:: iter:: once;
193
245
194
- use super :: collect_name_symbols;
246
+ use super :: { MangleOptionsKeepNames , collect_name_symbols} ;
195
247
196
- fn collect ( source_text : & str ) -> FxHashSet < String > {
248
+ fn collect ( opts : MangleOptionsKeepNames , source_text : & str ) -> FxHashSet < String > {
197
249
let allocator = Allocator :: default ( ) ;
198
250
let ret = Parser :: new ( & allocator, source_text, SourceType :: mjs ( ) ) . parse ( ) ;
199
251
assert ! ( !ret. panicked, "{source_text}" ) ;
200
252
assert ! ( ret. errors. is_empty( ) , "{source_text}" ) ;
201
253
let ret = SemanticBuilder :: new ( ) . build ( & ret. program ) ;
202
254
assert ! ( ret. errors. is_empty( ) , "{source_text}" ) ;
203
255
let semantic = ret. semantic ;
204
- let symbols = collect_name_symbols ( semantic. scoping ( ) , semantic. nodes ( ) ) ;
256
+ let symbols = collect_name_symbols ( opts , semantic. scoping ( ) , semantic. nodes ( ) ) ;
205
257
symbols
206
258
. into_iter ( )
207
259
. map ( |symbol_id| semantic. scoping ( ) . symbol_name ( symbol_id) . to_string ( ) )
208
260
. collect ( )
209
261
}
210
262
263
+ fn data ( s : & str ) -> FxHashSet < String > {
264
+ FxHashSet :: from_iter ( [ s. to_string ( ) ] )
265
+ }
266
+
267
+ fn function_only ( ) -> MangleOptionsKeepNames {
268
+ MangleOptionsKeepNames { function : true , class : false }
269
+ }
270
+
271
+ fn class_only ( ) -> MangleOptionsKeepNames {
272
+ MangleOptionsKeepNames { function : false , class : true }
273
+ }
274
+
211
275
#[ test]
212
276
fn test_declarations ( ) {
213
- assert_eq ! ( collect( "function foo() {}" ) , once ( "foo" . to_string ( ) ) . collect ( ) ) ;
214
- assert_eq ! ( collect( "class Foo {}" ) , once ( "Foo" . to_string ( ) ) . collect ( ) ) ;
277
+ assert_eq ! ( collect( function_only ( ) , "function foo() {}" ) , data ( "foo" ) ) ;
278
+ assert_eq ! ( collect( class_only ( ) , "class Foo {}" ) , data ( "Foo" ) ) ;
215
279
}
216
280
217
281
#[ test]
218
282
fn test_simple_declare_init ( ) {
219
- assert_eq ! ( collect( "var foo = function() {}" ) , once ( "foo" . to_string ( ) ) . collect ( ) ) ;
220
- assert_eq ! ( collect( "var foo = () => {}" ) , once ( "foo" . to_string ( ) ) . collect ( ) ) ;
221
- assert_eq ! ( collect( "var Foo = class {}" ) , once ( "Foo" . to_string ( ) ) . collect ( ) ) ;
283
+ assert_eq ! ( collect( function_only ( ) , "var foo = function() {}" ) , data ( "foo" ) ) ;
284
+ assert_eq ! ( collect( function_only ( ) , "var foo = () => {}" ) , data ( "foo" ) ) ;
285
+ assert_eq ! ( collect( class_only ( ) , "var Foo = class {}" ) , data ( "Foo" ) ) ;
222
286
}
223
287
224
288
#[ test]
225
289
fn test_simple_assign ( ) {
226
- assert_eq ! ( collect( "var foo; foo = function() {}" ) , once ( "foo" . to_string ( ) ) . collect ( ) ) ;
227
- assert_eq ! ( collect( "var foo; foo = () => {}" ) , once ( "foo" . to_string ( ) ) . collect ( ) ) ;
228
- assert_eq ! ( collect( "var Foo; Foo = class {}" ) , once ( "Foo" . to_string ( ) ) . collect ( ) ) ;
290
+ assert_eq ! ( collect( function_only ( ) , "var foo; foo = function() {}" ) , data ( "foo" ) ) ;
291
+ assert_eq ! ( collect( function_only ( ) , "var foo; foo = () => {}" ) , data ( "foo" ) ) ;
292
+ assert_eq ! ( collect( class_only ( ) , "var Foo; Foo = class {}" ) , data ( "Foo" ) ) ;
229
293
230
- assert_eq ! ( collect( "var foo; foo ||= function() {}" ) , once( "foo" . to_string( ) ) . collect( ) ) ;
231
- assert_eq ! (
232
- collect( "var foo = 1; foo &&= function() {}" ) ,
233
- once( "foo" . to_string( ) ) . collect( )
234
- ) ;
235
- assert_eq ! ( collect( "var foo; foo ??= function() {}" ) , once( "foo" . to_string( ) ) . collect( ) ) ;
294
+ assert_eq ! ( collect( function_only( ) , "var foo; foo ||= function() {}" ) , data( "foo" ) ) ;
295
+ assert_eq ! ( collect( function_only( ) , "var foo = 1; foo &&= function() {}" ) , data( "foo" ) ) ;
296
+ assert_eq ! ( collect( function_only( ) , "var foo; foo ??= function() {}" ) , data( "foo" ) ) ;
236
297
}
237
298
238
299
#[ test]
239
300
fn test_default_declarations ( ) {
240
- assert_eq ! ( collect( "var [foo = function() {}] = []" ) , once ( "foo" . to_string ( ) ) . collect ( ) ) ;
241
- assert_eq ! ( collect( "var [foo = () => {}] = []" ) , once ( "foo" . to_string ( ) ) . collect ( ) ) ;
242
- assert_eq ! ( collect( "var [Foo = class {}] = []" ) , once ( "Foo" . to_string ( ) ) . collect ( ) ) ;
243
- assert_eq ! ( collect( "var { foo = function() {} } = {}" ) , once ( "foo" . to_string ( ) ) . collect ( ) ) ;
301
+ assert_eq ! ( collect( function_only ( ) , "var [foo = function() {}] = []" ) , data ( "foo" ) ) ;
302
+ assert_eq ! ( collect( function_only ( ) , "var [foo = () => {}] = []" ) , data ( "foo" ) ) ;
303
+ assert_eq ! ( collect( class_only ( ) , "var [Foo = class {}] = []" ) , data ( "Foo" ) ) ;
304
+ assert_eq ! ( collect( function_only ( ) , "var { foo = function() {} } = {}" ) , data ( "foo" ) ) ;
244
305
}
245
306
246
307
#[ test]
247
308
fn test_default_assign ( ) {
309
+ assert_eq ! ( collect( function_only( ) , "var foo; [foo = function() {}] = []" ) , data( "foo" ) ) ;
310
+ assert_eq ! ( collect( function_only( ) , "var foo; [foo = () => {}] = []" ) , data( "foo" ) ) ;
311
+ assert_eq ! ( collect( class_only( ) , "var Foo; [Foo = class {}] = []" ) , data( "Foo" ) ) ;
248
312
assert_eq ! (
249
- collect( "var foo; [foo = function() {}] = []" ) ,
250
- once( "foo" . to_string( ) ) . collect( )
251
- ) ;
252
- assert_eq ! ( collect( "var foo; [foo = () => {}] = []" ) , once( "foo" . to_string( ) ) . collect( ) ) ;
253
- assert_eq ! ( collect( "var Foo; [Foo = class {}] = []" ) , once( "Foo" . to_string( ) ) . collect( ) ) ;
254
- assert_eq ! (
255
- collect( "var foo; ({ foo = function() {} } = {})" ) ,
256
- once( "foo" . to_string( ) ) . collect( )
313
+ collect( function_only( ) , "var foo; ({ foo = function() {} } = {})" ) ,
314
+ data( "foo" )
257
315
) ;
258
316
}
259
317
260
318
#[ test]
261
319
fn test_for_in_declaration ( ) {
262
- assert_eq ! (
263
- collect( "for (var foo = function() {} in []) {}" ) ,
264
- once( "foo" . to_string( ) ) . collect( )
265
- ) ;
266
- assert_eq ! ( collect( "for (var foo = () => {} in []) {}" ) , once( "foo" . to_string( ) ) . collect( ) ) ;
267
- assert_eq ! ( collect( "for (var Foo = class {} in []) {}" ) , once( "Foo" . to_string( ) ) . collect( ) ) ;
320
+ assert_eq ! ( collect( function_only( ) , "for (var foo = function() {} in []) {}" ) , data( "foo" ) ) ;
321
+ assert_eq ! ( collect( function_only( ) , "for (var foo = () => {} in []) {}" ) , data( "foo" ) ) ;
322
+ assert_eq ! ( collect( class_only( ) , "for (var Foo = class {} in []) {}" ) , data( "Foo" ) ) ;
268
323
}
269
324
}
0 commit comments