Skip to content

Commit

Permalink
fix(transformer/class-properties): fix symbol clashes in instance pro…
Browse files Browse the repository at this point in the history
…p initializers
  • Loading branch information
overlookmotel authored and Boshen committed Dec 14, 2024
1 parent 53e2bc0 commit 455c4a1
Show file tree
Hide file tree
Showing 6 changed files with 188 additions and 76 deletions.
11 changes: 8 additions & 3 deletions crates/oxc_transformer/src/es2022/class_properties/class.rs
Original file line number Diff line number Diff line change
Expand Up @@ -409,14 +409,18 @@ impl<'a, 'ctx> ClassProperties<'a, 'ctx> {
None
} else if let Some(constructor) = constructor {
// Existing constructor
// TODO: Set `self.instance_inits_constructor_scope_id = None` if constructor has no bindings
let constructor = constructor.value.as_mut();
if class.super_class.is_some() {
let (instance_inits_scope_id, insert_location) =
let (insert_scopes, insert_location) =
Self::replace_super_in_constructor(constructor, ctx);
self.instance_inits_scope_id = instance_inits_scope_id;
self.instance_inits_scope_id = insert_scopes.insert_in_scope_id;
self.instance_inits_constructor_scope_id = insert_scopes.constructor_scope_id;
Some(insert_location)
} else {
self.instance_inits_scope_id = constructor.scope_id();
let constructor_scope_id = constructor.scope_id();
self.instance_inits_scope_id = constructor_scope_id;
self.instance_inits_constructor_scope_id = Some(constructor_scope_id);
Some(InstanceInitsInsertLocation::ExistingConstructor(0))
}
} else {
Expand All @@ -427,6 +431,7 @@ impl<'a, 'ctx> ClassProperties<'a, 'ctx> {
ScopeFlags::Function | ScopeFlags::Constructor | ScopeFlags::StrictMode,
);
self.instance_inits_scope_id = constructor_scope_id;
self.instance_inits_constructor_scope_id = None;
Some(InstanceInitsInsertLocation::NewConstructor)
};

Expand Down
148 changes: 125 additions & 23 deletions crates/oxc_transformer/src/es2022/class_properties/constructor.rs
Original file line number Diff line number Diff line change
Expand Up @@ -99,13 +99,15 @@
//! ESBuild does not handle `super()` in constructor params correctly:
//! [ESBuild REPL](https://esbuild.github.io/try/#dAAwLjI0LjAALS10YXJnZXQ9ZXMyMDIwAGNsYXNzIEMgZXh0ZW5kcyBTIHsKICBwcm9wID0gZm9vKCk7CiAgY29uc3RydWN0b3IoeCA9IHN1cGVyKCksIHkgPSBzdXBlcigpKSB7fQp9Cg)
use rustc_hash::FxHashMap;

use oxc_allocator::Vec as ArenaVec;
use oxc_ast::{ast::*, visit::walk_mut, VisitMut, NONE};
use oxc_span::SPAN;
use oxc_syntax::{
node::NodeId,
scope::{ScopeFlags, ScopeId},
symbol::SymbolFlags,
symbol::{SymbolFlags, SymbolId},
};
use oxc_traverse::{BoundIdentifier, TraverseCtx};

Expand All @@ -126,11 +128,21 @@ pub(super) enum InstanceInitsInsertLocation<'a> {
SuperFnOutsideClass(BoundIdentifier<'a>),
}

/// Scopes related to inserting and transforming instance property initializers
pub(super) struct InstanceInitScopes {
/// Scope that instance prop initializers will be inserted into
pub insert_in_scope_id: ScopeId,
/// Scope of class constructor, if initializers will be inserted into constructor
/// (either directly, or in `_super` function within constructor)
pub constructor_scope_id: Option<ScopeId>,
}

impl<'a, 'ctx> ClassProperties<'a, 'ctx> {
/// Replace `super()` call(s) in constructor, if required.
///
/// Returns `InstanceInitsInsertLocation` detailing where instance property initializers
/// should be inserted.
/// Returns:
/// * `InstanceInitScopes` details the `ScopeId`s required for transforming instance property initializers.
/// * `InstanceInitsInsertLocation` detailing where instance property initializers should be inserted.
///
/// * `super()` first appears as a top level statement in constructor body (common case):
/// * Do not alter constructor.
Expand All @@ -151,14 +163,15 @@ impl<'a, 'ctx> ClassProperties<'a, 'ctx> {
///
/// See doc comment at top of this file for more details of last 3 cases.
///
/// If a `_super` function is required, binding for `_super`, and `ScopeId` of `_super` function
/// are recorded in the returned `InstanceInitsInsertLocation`.
/// If a `_super` function is required, binding for `_super` is recorded in the returned
/// `InstanceInitsInsertLocation`, and `ScopeId` from `_super` function is returned as
/// `insert_in_scope_id` in returned `InstanceInitScopes`.
///
/// This function does not create the `_super` function or insert it. That happens later.
pub(super) fn replace_super_in_constructor(
constructor: &mut Function<'a>,
ctx: &mut TraverseCtx<'a>,
) -> (ScopeId, InstanceInitsInsertLocation<'a>) {
) -> (InstanceInitScopes, InstanceInitsInsertLocation<'a>) {
// Find any `super()`s in constructor params and replace with `_super.call(super())`
let replacer = ConstructorParamsSuperReplacer::new(ctx);
if let Some(result) = replacer.replace(constructor) {
Expand All @@ -172,7 +185,7 @@ impl<'a, 'ctx> ClassProperties<'a, 'ctx> {
replacer.replace(body_stmts)
}

/// Insert property initializers into existing class constructor.
/// Insert instance property initializers.
///
/// `scope_id` has different meaning depending on type of `insertion_location`.
pub(super) fn insert_instance_inits(
Expand All @@ -193,7 +206,7 @@ impl<'a, 'ctx> ClassProperties<'a, 'ctx> {
Self::insert_constructor(class, scope_id, inits, ctx);
}
InstanceInitsInsertLocation::ExistingConstructor(stmt_index) => {
Self::insert_inits_into_constructor_as_statements(
self.insert_inits_into_constructor_as_statements(
class,
inits,
constructor_index,
Expand All @@ -202,7 +215,7 @@ impl<'a, 'ctx> ClassProperties<'a, 'ctx> {
);
}
InstanceInitsInsertLocation::SuperFnInsideConstructor(super_binding) => {
Self::create_super_function_inside_constructor(
self.create_super_function_inside_constructor(
class,
inits,
super_binding,
Expand Down Expand Up @@ -282,27 +295,38 @@ impl<'a, 'ctx> ClassProperties<'a, 'ctx> {

/// Insert instance property initializers into constructor body at `insertion_index`.
fn insert_inits_into_constructor_as_statements(
&mut self,
class: &mut Class<'a>,
inits: Vec<Expression<'a>>,
constructor_index: usize,
insertion_index: usize,
ctx: &TraverseCtx<'a>,
ctx: &mut TraverseCtx<'a>,
) {
let body_stmts = Self::get_constructor_body_stmts(class, constructor_index);
// Rename any symbols in constructor which clash with references in inits
let constructor = Self::get_constructor(class, constructor_index);
self.rename_clashing_symbols(constructor, ctx);

// Insert inits into constructor body
let body_stmts = &mut constructor.body.as_mut().unwrap().statements;
body_stmts.splice(insertion_index..insertion_index, exprs_into_stmts(inits, ctx));
}

/// Create `_super` function containing instance property initializers,
/// and insert at top of constructor body.
/// `var _super = (..._args) => (super(..._args), <inits>, this);`
fn create_super_function_inside_constructor(
&mut self,
class: &mut Class<'a>,
inits: Vec<Expression<'a>>,
super_binding: &BoundIdentifier<'a>,
super_func_scope_id: ScopeId,
constructor_index: usize,
ctx: &mut TraverseCtx<'a>,
) {
// Rename any symbols in constructor which clash with references in inits
let constructor = Self::get_constructor(class, constructor_index);
self.rename_clashing_symbols(constructor, ctx);

// `(super(..._args), <inits>, this)`
//
// TODO(improve-on-babel): When not in loose mode, inits are `_defineProperty(this, propName, value)`.
Expand Down Expand Up @@ -357,7 +381,7 @@ impl<'a, 'ctx> ClassProperties<'a, 'ctx> {
));

// Insert at top of function
let body_stmts = Self::get_constructor_body_stmts(class, constructor_index);
let body_stmts = &mut constructor.body.as_mut().unwrap().statements;
body_stmts.insert(0, super_func_decl);
}

Expand Down Expand Up @@ -420,17 +444,47 @@ impl<'a, 'ctx> ClassProperties<'a, 'ctx> {
self.ctx.var_declarations.insert_let(super_binding, init, ctx);
}

/// Rename any symbols in constructor which clash with symbols used in initializers
fn rename_clashing_symbols(
&mut self,
constructor: &mut Function<'a>,
ctx: &mut TraverseCtx<'a>,
) {
if self.clashing_constructor_symbols.is_empty() {
return;
}

// Rename symbols to UIDs
let constructor_scope_id = constructor.scope_id();
for (&symbol_id, name) in &mut self.clashing_constructor_symbols {
// Generate replacement UID name
let new_name = ctx.generate_uid_name(name);
// Save replacement name in `clashing_constructor_symbols`
*name = ctx.ast.atom(&new_name);
// Rename symbol and binding
ctx.rename_symbol(symbol_id, constructor_scope_id, new_name);
}

// Rename identifiers for clashing symbols in constructor params and body
let mut renamer =
ConstructorSymbolRenamer::new(&mut self.clashing_constructor_symbols, ctx);
renamer.visit_function(constructor, ScopeFlags::empty());

// Empty `clashing_constructor_symbols` hashmap for reuse on next class
self.clashing_constructor_symbols.clear();
}

/// Get body statements of constructor, given constructor's index within class elements.
fn get_constructor_body_stmts<'b>(
fn get_constructor<'b>(
class: &'b mut Class<'a>,
constructor_index: usize,
) -> &'b mut ArenaVec<'a, Statement<'a>> {
) -> &'b mut Function<'a> {
let constructor = match class.body.body.get_mut(constructor_index) {
Some(ClassElement::MethodDefinition(constructor)) => constructor.as_mut(),
_ => unreachable!(),
};
debug_assert!(constructor.kind == MethodDefinitionKind::Constructor);
&mut constructor.value.body.as_mut().unwrap().statements
&mut constructor.value
}
}

Expand All @@ -450,7 +504,7 @@ impl<'a, 'c> ConstructorParamsSuperReplacer<'a, 'c> {
fn replace(
mut self,
constructor: &mut Function<'a>,
) -> Option<(ScopeId, InstanceInitsInsertLocation<'a>)> {
) -> Option<(InstanceInitScopes, InstanceInitsInsertLocation<'a>)> {
self.visit_formal_parameters(&mut constructor.params);

#[expect(clippy::question_mark)]
Expand Down Expand Up @@ -479,8 +533,12 @@ impl<'a, 'c> ConstructorParamsSuperReplacer<'a, 'c> {
NodeId::DUMMY,
ScopeFlags::Function | ScopeFlags::StrictMode,
);
let insert_scopes = InstanceInitScopes {
insert_in_scope_id: super_func_scope_id,
constructor_scope_id: None,
};

Some((super_func_scope_id, insert_location))
Some((insert_scopes, insert_location))
}
}

Expand Down Expand Up @@ -595,20 +653,24 @@ impl<'a, 'c> ConstructorBodySuperReplacer<'a, 'c> {
fn replace(
mut self,
body_stmts: &mut ArenaVec<'a, Statement<'a>>,
) -> (ScopeId, InstanceInitsInsertLocation<'a>) {
) -> (InstanceInitScopes, InstanceInitsInsertLocation<'a>) {
let mut body_stmts_iter = body_stmts.iter_mut();

loop {
let mut body_stmts_iter_enumerated = body_stmts_iter.by_ref().enumerate();
if let Some((index, stmt)) = body_stmts_iter_enumerated.next() {
// If statement is standalone `super()`, insert inits after `super()`.
// We can avoid a nested `_super` function for this common case.
// We can avoid a `_super` function for this common case.
if let Statement::ExpressionStatement(expr_stmt) = &*stmt {
if let Expression::CallExpression(call_expr) = &expr_stmt.expression {
if call_expr.callee.is_super() {
let insert_location =
InstanceInitsInsertLocation::ExistingConstructor(index + 1);
return (self.constructor_scope_id, insert_location);
let insert_scopes = InstanceInitScopes {
insert_in_scope_id: self.constructor_scope_id,
constructor_scope_id: Some(self.constructor_scope_id),
};
return (insert_scopes, insert_location);
}
}
}
Expand All @@ -630,10 +692,14 @@ impl<'a, 'c> ConstructorBodySuperReplacer<'a, 'c> {
// could be. But this should very rarely happen in practice, and minifier will delete
// the `_super` function as dead code.
// TODO: Delete the initializers instead.
let insert_scopes = InstanceInitScopes {
insert_in_scope_id: self.create_super_func_scope(),
constructor_scope_id: Some(self.constructor_scope_id),
};
let super_binding = self.create_super_binding();
let insert_location =
InstanceInitsInsertLocation::SuperFnInsideConstructor(super_binding);
return (self.create_super_func_scope(), insert_location);
return (insert_scopes, insert_location);
}
}

Expand All @@ -643,10 +709,13 @@ impl<'a, 'c> ConstructorBodySuperReplacer<'a, 'c> {
self.visit_statement(stmt);
}

let super_func_scope_id = self.create_super_func_scope();
let insert_scopes = InstanceInitScopes {
insert_in_scope_id: self.create_super_func_scope(),
constructor_scope_id: Some(self.constructor_scope_id),
};
let super_binding = self.super_binding.unwrap();
let insert_location = InstanceInitsInsertLocation::SuperFnInsideConstructor(super_binding);
(super_func_scope_id, insert_location)
(insert_scopes, insert_location)
}

/// Create scope for `_super` function inside constructor body
Expand Down Expand Up @@ -737,6 +806,39 @@ impl<'a, 'c> ConstructorBodySuperReplacer<'a, 'c> {
}
}

/// Visitor to rename bindings and references.
struct ConstructorSymbolRenamer<'a, 'v> {
clashing_symbols: &'v mut FxHashMap<SymbolId, Atom<'a>>,
ctx: &'v TraverseCtx<'a>,
}

impl<'a, 'v> ConstructorSymbolRenamer<'a, 'v> {
fn new(
clashing_symbols: &'v mut FxHashMap<SymbolId, Atom<'a>>,
ctx: &'v TraverseCtx<'a>,
) -> Self {
Self { clashing_symbols, ctx }
}
}

impl<'a, 'v> VisitMut<'a> for ConstructorSymbolRenamer<'a, 'v> {
fn visit_binding_identifier(&mut self, ident: &mut BindingIdentifier<'a>) {
let symbol_id = ident.symbol_id();
if let Some(new_name) = self.clashing_symbols.get(&symbol_id) {
ident.name = new_name.clone();
}
}

fn visit_identifier_reference(&mut self, ident: &mut IdentifierReference<'a>) {
let reference_id = ident.reference_id();
if let Some(symbol_id) = self.ctx.symbols().get_reference(reference_id).symbol_id() {
if let Some(new_name) = self.clashing_symbols.get(&symbol_id) {
ident.name = new_name.clone();
}
}
}
}

/// `super(...args);`
fn create_super_call<'a>(
args_binding: &BoundIdentifier<'a>,
Expand Down
Loading

0 comments on commit 455c4a1

Please sign in to comment.