use oxc::{
allocator::TakeIn,
ast::{NONE, ast},
span::SPAN,
};
use oxc_traverse::Traverse;
use rolldown_ecmascript::{
CJS_EXPORTS_REF_STR, CJS_MODULE_REF_STR, CJS_ROLLDOWN_EXPORTS_REF,
CJS_ROLLDOWN_EXPORTS_REF_IDENT, CJS_ROLLDOWN_MODULE_REF_IDENT,
};
use crate::hmr::{
hmr_ast_finalizer::HmrAstFinalizer,
utils::{HmrAstBuilder, MODULE_ID_PARAM_FOR_HMR},
};
impl<'ast> Traverse<'ast, ()> for HmrAstFinalizer<'_, 'ast> {
fn enter_program(
&mut self,
node: &mut ast::Program<'ast>,
ctx: &mut oxc_traverse::TraverseCtx<'ast, ()>,
) {
let taken_body = node.body.take_in(self.alloc);
node.body.reserve_exact(taken_body.len());
taken_body.into_iter().for_each(|top_level_stmt| {
self.handle_top_level_stmt(&mut node.body, top_level_stmt, ctx.scoping());
});
}
fn exit_program(
&mut self,
node: &mut ast::Program<'ast>,
ctx: &mut oxc_traverse::TraverseCtx<'ast, ()>,
) {
let mut try_block =
self.snippet.builder.alloc_block_statement(SPAN, self.snippet.builder.vec());
let dependencies_init_fn_stmts: Vec<_> = self
.dependencies
.iter()
.filter_map(|dep| self.affected_module_idx_to_init_fn_name.get(dep))
.map(|fn_name| {
let call_expr = self.snippet.call_expr_expr(fn_name);
self.snippet.builder.statement_expression(SPAN, call_expr)
})
.collect();
let runtime_module_register = self.generate_runtime_module_register_for_hmr(ctx.scoping());
try_block.body.reserve_exact(
runtime_module_register.len() + node.body.len() + dependencies_init_fn_stmts.len() + 1 ,
);
try_block.body.extend(runtime_module_register);
try_block.body.extend(dependencies_init_fn_stmts);
try_block.body.push(self.create_module_hot_context_initializer_stmt());
try_block.body.extend(node.body.take_in(self.alloc));
node
.body
.extend(std::mem::take(&mut self.generated_static_import_stmts_from_external).into_values());
let final_block = self.snippet.builder.alloc_block_statement(SPAN, self.snippet.builder.vec());
let try_stmt =
self.snippet.builder.alloc_try_statement(SPAN, try_block, NONE, Some(final_block));
let init_fn_name = &self.affected_module_idx_to_init_fn_name[&self.module.idx];
let module_id_param = self.snippet.builder.formal_parameter(
SPAN,
self.builder.vec(),
self.snippet.builder.binding_pattern_binding_identifier(SPAN, MODULE_ID_PARAM_FOR_HMR),
NONE,
NONE,
false,
None,
false,
false,
);
let params = self.snippet.builder.formal_parameters(
SPAN,
ast::FormalParameterKind::Signature,
{
if self.module.exports_kind.is_commonjs() {
self.snippet.builder.vec_from_array([
self.snippet.builder.formal_parameter(
SPAN,
self.builder.vec(),
self
.snippet
.builder
.binding_pattern_binding_identifier(SPAN, CJS_ROLLDOWN_EXPORTS_REF_IDENT),
NONE,
NONE,
false,
None,
false,
false,
),
self.snippet.builder.formal_parameter(
SPAN,
self.builder.vec(),
self
.snippet
.builder
.binding_pattern_binding_identifier(SPAN, CJS_ROLLDOWN_MODULE_REF_IDENT),
NONE,
NONE,
false,
None,
false,
false,
),
module_id_param,
])
} else {
self.snippet.builder.vec1(module_id_param)
}
},
NONE,
);
let mut user_code_wrapper = self.snippet.builder.alloc_function(
SPAN,
ast::FunctionType::FunctionExpression,
None,
false,
false,
false,
NONE,
NONE,
params,
NONE,
Some(self.snippet.builder.function_body(
SPAN,
self.snippet.builder.vec(),
self.snippet.builder.vec1(ast::Statement::TryStatement(try_stmt)),
)),
);
user_code_wrapper.pife = self.use_pife_for_module_wrappers;
let mut initializer_args = self.snippet.builder.vec_with_capacity(3);
initializer_args.push(ast::Argument::StringLiteral(self.snippet.builder.alloc_string_literal(
SPAN,
self.snippet.builder.str(&self.module.stable_id),
None,
)));
initializer_args
.push(ast::Argument::from(ast::Expression::FunctionExpression(user_code_wrapper)));
if self.dedup_module_initializer {
initializer_args.push(ast::Argument::from(self.snippet.builder.expression_numeric_literal(
SPAN,
1.0,
None,
ast::NumberBase::Decimal,
)));
}
let initializer_callee = if self.module.exports_kind.is_commonjs() {
"__rolldown_runtime__.createCjsInitializer"
} else {
"__rolldown_runtime__.createEsmInitializer"
};
let initializer_call = self.snippet.builder.alloc_call_expression(
SPAN,
self.snippet.id_ref_expr(initializer_callee, SPAN),
NONE,
initializer_args,
false,
);
let var_decl = self.snippet.builder.alloc_variable_declaration(
SPAN,
ast::VariableDeclarationKind::Var,
self.snippet.builder.vec1(
self.snippet.builder.variable_declarator(
SPAN,
ast::VariableDeclarationKind::Var,
self
.snippet
.builder
.binding_pattern_binding_identifier(SPAN, self.snippet.builder.str(init_fn_name)),
NONE,
Some(ast::Expression::CallExpression(initializer_call)),
false,
),
),
false,
);
node.body.push(ast::Statement::VariableDeclaration(var_decl));
}
fn enter_call_expression(
&mut self,
node: &mut ast::CallExpression<'ast>,
_ctx: &mut oxc_traverse::TraverseCtx<'ast, ()>,
) {
self.rewrite_hot_accept_call_deps(node);
}
fn exit_expression(
&mut self,
node: &mut oxc::ast::ast::Expression<'ast>,
ctx: &mut oxc_traverse::TraverseCtx<'ast, ()>,
) {
if let ast::Expression::ThisExpression(this_expr) = node
&& self.module.exports_kind.is_commonjs()
&& self.module.ecma_view.this_expr_replace_map.contains_key(&this_expr.span)
{
*node = self.snippet.id_ref_expr(CJS_ROLLDOWN_EXPORTS_REF, SPAN);
return;
}
self.try_rewrite_dynamic_import(node);
self.try_rewrite_require(node, ctx);
self.rewrite_import_meta_hot(node);
}
fn exit_identifier_reference(
&mut self,
node: &mut ast::IdentifierReference<'ast>,
ctx: &mut oxc_traverse::TraverseCtx<'ast, ()>,
) {
self.rewrite_identifier_reference(node, ctx);
}
}
impl<'ast> HmrAstFinalizer<'_, 'ast> {
fn rewrite_identifier_reference(
&self,
ident: &mut ast::IdentifierReference<'ast>,
ctx: &oxc_traverse::TraverseCtx<'ast, ()>,
) {
let Some(reference_id) = ident.reference_id.get() else {
return;
};
let reference = ctx.scoping().get_reference(reference_id);
if let Some(symbol_id) = reference.symbol_id() {
if let Some(binding_name) = self.import_bindings.get(&symbol_id) {
ident.name = self.snippet.atom(binding_name.as_str()).into();
}
} else if ident.name == CJS_EXPORTS_REF_STR {
ident.name = CJS_ROLLDOWN_EXPORTS_REF_IDENT;
} else if ident.name == CJS_MODULE_REF_STR {
ident.name = CJS_ROLLDOWN_MODULE_REF_IDENT;
}
}
}