oxc_transformer 0.136.0

A collection of JavaScript tools written in Rust.
Documentation
//! Utility transform to add new statements before or after the specified statement.
//!
//! `StatementInjectorStore` contains a `FxHashMap<Address, Vec<AdjacentStatement>>`. It is stored on `TransformState`.
//!
//! `Common` transform's `exit_statements` calls `insert_into_statements` to insert new statements
//! before or after a statement which is determined by the address of the statement.
//!
//! Other transforms can add statements to the store with following methods:
//!
//! ```rs
//! ctx.state.statement_injector.insert_before(address, statement);
//! ctx.state.statement_injector.insert_after(address, statement);
//! ctx.state.statement_injector.insert_many_after(address, statements);
//! ```

use std::collections::hash_map::Entry;

use rustc_hash::FxHashMap;

use oxc_allocator::{Address, GetAddress, Vec as ArenaVec};
use oxc_ast::{AstBuilder, ast::*};

/// Store for statements to be added to the statements.
pub struct StatementInjectorStore<'a> {
    insertions: FxHashMap<Address, Vec<AdjacentStatement<'a>>>,
}

#[derive(Debug)]
enum Direction {
    Before,
    After,
}

#[derive(Debug)]
struct AdjacentStatement<'a> {
    stmt: Statement<'a>,
    direction: Direction,
}

// Public methods
impl StatementInjectorStore<'_> {
    /// Create new `StatementInjectorStore`.
    pub fn new() -> Self {
        Self { insertions: FxHashMap::default() }
    }
}

// Insertion methods.
//
// Each of these functions is split into 2 parts:
//
// 1. Public outer function which is generic over any `GetAddress`.
// 2. Private inner function which is non-generic and takes `Address`.
//
// Outer functions are marked `#[inline]`, as `GetAddress::address` is generally only 1 or 2 instructions.
// The non-trivial inner functions are not marked `#[inline]` - compiler can decide whether to inline or not.
impl<'a> StatementInjectorStore<'a> {
    /// Add a statement to be inserted immediately before the target statement.
    #[inline]
    pub fn insert_before<A: GetAddress>(&mut self, target: &A, stmt: Statement<'a>) {
        self.insert_before_address(target.address(), stmt);
    }

    fn insert_before_address(&mut self, target: Address, stmt: Statement<'a>) {
        let adjacent_stmts = self.insertions.entry(target).or_default();
        let index = adjacent_stmts
            .iter()
            .position(|s| matches!(s.direction, Direction::After))
            .unwrap_or(adjacent_stmts.len());
        adjacent_stmts.insert(index, AdjacentStatement { stmt, direction: Direction::Before });
    }

    /// Add a statement to be inserted immediately after the target statement.
    #[inline]
    pub fn insert_after<A: GetAddress>(&mut self, target: &A, stmt: Statement<'a>) {
        self.insert_after_address(target.address(), stmt);
    }

    fn insert_after_address(&mut self, target: Address, stmt: Statement<'a>) {
        let adjacent_stmts = self.insertions.entry(target).or_default();
        adjacent_stmts.push(AdjacentStatement { stmt, direction: Direction::After });
    }

    /// Add multiple statements to be inserted immediately before the target statement.
    #[inline]
    pub fn insert_many_before<A, S>(&mut self, target: &A, stmts: S)
    where
        A: GetAddress,
        S: IntoIterator<Item = Statement<'a>>,
    {
        self.insert_many_before_address(target.address(), stmts);
    }

    fn insert_many_before_address<S>(&mut self, target: Address, stmts: S)
    where
        S: IntoIterator<Item = Statement<'a>>,
    {
        let adjacent_stmts = self.insertions.entry(target).or_default();
        adjacent_stmts.splice(
            0..0,
            stmts.into_iter().map(|stmt| AdjacentStatement { stmt, direction: Direction::Before }),
        );
    }

    /// Add multiple statements to be inserted immediately after the target statement.
    #[inline]
    pub fn insert_many_after<A, S>(&mut self, target: &A, stmts: S)
    where
        A: GetAddress,
        S: IntoIterator<Item = Statement<'a>>,
    {
        self.insert_many_after_address(target.address(), stmts);
    }

    fn insert_many_after_address<S>(&mut self, target: Address, stmts: S)
    where
        S: IntoIterator<Item = Statement<'a>>,
    {
        let adjacent_stmts = self.insertions.entry(target).or_default();
        adjacent_stmts.extend(
            stmts.into_iter().map(|stmt| AdjacentStatement { stmt, direction: Direction::After }),
        );
    }

    /// Move insertions from one [`Address`] to another.
    ///
    /// Use this if you convert one statement to another, and other code may have attached
    /// insertions to the original statement.
    #[inline]
    pub fn move_insertions<A1: GetAddress, A2: GetAddress>(
        &mut self,
        old_target: &A1,
        new_target: &A2,
    ) {
        self.move_insertions_address(old_target.address(), new_target.address());
    }

    fn move_insertions_address(&mut self, old_address: Address, new_address: Address) {
        let Some(mut adjacent_stmts) = self.insertions.remove(&old_address) else { return };

        match self.insertions.entry(new_address) {
            Entry::Occupied(entry) => {
                entry.into_mut().append(&mut adjacent_stmts);
            }
            Entry::Vacant(entry) => {
                entry.insert(adjacent_stmts);
            }
        }
    }
}

// Internal methods - called by `Common` transform
impl<'a> StatementInjectorStore<'a> {
    /// Insert statements immediately before / after the target statement.
    pub(crate) fn insert_into_statements(
        &mut self,
        statements: &mut ArenaVec<'a, Statement<'a>>,
        ast: AstBuilder<'a>,
    ) {
        if self.insertions.is_empty() {
            return;
        }

        let new_statement_count = statements
            .iter()
            .filter_map(|s| self.insertions.get(&s.address()).map(Vec::len))
            .sum::<usize>();
        if new_statement_count == 0 {
            return;
        }

        let mut new_statements = ast.vec_with_capacity(statements.len() + new_statement_count);

        for stmt in statements.drain(..) {
            match self.insertions.remove(&stmt.address()) {
                Some(mut adjacent_stmts) => {
                    let first_after_stmt_index = adjacent_stmts
                        .iter()
                        .position(|s| matches!(s.direction, Direction::After))
                        .unwrap_or(adjacent_stmts.len());
                    if first_after_stmt_index != 0 {
                        let right = adjacent_stmts.split_off(first_after_stmt_index);
                        new_statements.extend(adjacent_stmts.into_iter().map(|s| s.stmt));
                        new_statements.push(stmt);
                        new_statements.extend(right.into_iter().map(|s| s.stmt));
                    } else {
                        new_statements.push(stmt);
                        new_statements.extend(adjacent_stmts.into_iter().map(|s| s.stmt));
                    }
                }
                _ => {
                    new_statements.push(stmt);
                }
            }
        }

        *statements = new_statements;
    }

    // Assertion for checking if no remaining insertions are left.
    // `#[inline(always)]` because this is a no-op in release mode
    #[expect(clippy::inline_always)]
    #[inline(always)]
    pub(crate) fn assert_no_insertions_remaining(&self) {
        debug_assert!(self.insertions.is_empty());
    }
}