diff --git a/crates/oxc_transformer/src/es2018/async_generator_functions/for_await.rs b/crates/oxc_transformer/src/es2018/async_generator_functions/for_await.rs index eb146c4e2d0709..c3bb9a670d5b84 100644 --- a/crates/oxc_transformer/src/es2018/async_generator_functions/for_await.rs +++ b/crates/oxc_transformer/src/es2018/async_generator_functions/for_await.rs @@ -1,6 +1,6 @@ //! This module is responsible for transforming `for await` to `for` statement -use oxc_allocator::Vec; +use oxc_allocator::{CloneIn, GetAddress, Vec}; use oxc_ast::{ast::*, NONE}; use oxc_semantic::{ScopeFlags, ScopeId, SymbolFlags}; use oxc_span::SPAN; @@ -10,7 +10,57 @@ use super::AsyncGeneratorFunctions; use crate::{common::helper_loader::Helper, es2017::AsyncGeneratorExecutor}; impl<'a, 'ctx> AsyncGeneratorFunctions<'a, 'ctx> { - pub(crate) fn transform_for_of_statement( + pub(crate) fn transform_statement( + &mut self, + stmt: &mut Statement<'a>, + ctx: &mut TraverseCtx<'a>, + ) { + let (for_of, label) = match stmt { + Statement::LabeledStatement(labeled) => { + let LabeledStatement { label, body, .. } = labeled.as_mut(); + if let Statement::ForOfStatement(for_of) = body { + (for_of, Some(label)) + } else { + return; + } + } + Statement::ForOfStatement(for_of) => (for_of, None), + _ => return, + }; + + if !for_of.r#await { + return; + } + + // We need to replace the current statement with new statements, + // but we don't have a such method to do it, so we leverage the statement injector. + // + // Now, we use below steps to workaround it: + // 1. Use the last statement as the new statement. + // 2. insert the rest of the statements before the current statement. + // TODO: Once we have a method to replace the current statement, we can simplify this logic. + let mut statements = self.transform_for_of_statement(for_of, ctx); + let mut last_statement = statements.pop().unwrap(); + + // If it's a labeled statement, we need to wrap the ForStatement with a labeled statement. + if let Some(label) = label { + let Statement::TryStatement(try_statement) = &mut last_statement else { + unreachable!("The last statement should be a try statement, please see the `build_for_await` function"); + }; + let try_statement_block_body = &mut try_statement.block.body; + let for_statement = try_statement_block_body.pop().unwrap(); + try_statement_block_body.push(ctx.ast.statement_labeled( + SPAN, + label.clone_in(ctx.ast.allocator), + for_statement, + )); + } + + *stmt = last_statement; + self.ctx.statement_injector.insert_many_before(&stmt.address(), statements); + } + + pub(self) fn transform_for_of_statement( &mut self, stmt: &mut ForOfStatement<'a>, ctx: &mut TraverseCtx<'a>, @@ -66,14 +116,14 @@ impl<'a, 'ctx> AsyncGeneratorFunctions<'a, 'ctx> { statements }; - Self::build_for_await( - self.ctx.helper_load(Helper::AsyncIterator, ctx), - ctx.ast.move_expression(&mut stmt.right), - &step_key, - body, - stmt.scope_id(), + let iterator = ctx.ast.move_expression(&mut stmt.right); + let iterator = self.ctx.helper_call_expr( + Helper::AsyncIterator, + ctx.ast.vec1(Argument::from(iterator)), ctx, - ) + ); + let scope_id = stmt.scope_id(); + Self::build_for_await(iterator, &step_key, body, scope_id, ctx) } /// Build a `for` statement used to replace the `for await` statement. @@ -110,8 +160,7 @@ impl<'a, 'ctx> AsyncGeneratorFunctions<'a, 'ctx> { /// Based on Babel's implementation: /// fn build_for_await( - get_identifier: Expression<'a>, - object: Expression<'a>, + iterator: Expression<'a>, step_key: &BoundIdentifier<'a>, body: Vec<'a, Statement<'a>>, for_of_scope_id: ScopeId, @@ -185,13 +234,7 @@ impl<'a, 'ctx> AsyncGeneratorFunctions<'a, 'ctx> { SPAN, VariableDeclarationKind::Var, iterator_key.create_binding_pattern(ctx), - Some(ctx.ast.expression_call( - SPAN, - get_identifier, - NONE, - ctx.ast.vec1(Argument::from(object)), - false, - )), + Some(iterator), false, )); items.push(ctx.ast.variable_declarator( @@ -272,6 +315,12 @@ impl<'a, 'ctx> AsyncGeneratorFunctions<'a, 'ctx> { }, for_statement_scope_id, )); + + // // If has a label, we need to wrap the for statement with a labeled statement. + // // e.g. `label: for await (let i of test) {}` to `label: { for (let i of test) {} }` + // if let Some(label) = label { + // statement = ctx.ast.statement_labeled(SPAN, label, statement); + // } ctx.ast.block_statement_with_scope_id(SPAN, ctx.ast.vec1(for_statement), block_scope_id) }; diff --git a/crates/oxc_transformer/src/es2018/async_generator_functions/mod.rs b/crates/oxc_transformer/src/es2018/async_generator_functions/mod.rs index 81f5a32c7e0e32..f39a54b40cfb81 100644 --- a/crates/oxc_transformer/src/es2018/async_generator_functions/mod.rs +++ b/crates/oxc_transformer/src/es2018/async_generator_functions/mod.rs @@ -66,7 +66,6 @@ mod for_await; -use oxc_allocator::GetAddress; use oxc_ast::ast::*; use oxc_span::SPAN; use oxc_traverse::{Ancestor, Traverse, TraverseCtx}; @@ -109,23 +108,7 @@ impl<'a, 'ctx> Traverse<'a> for AsyncGeneratorFunctions<'a, 'ctx> { } fn enter_statement(&mut self, stmt: &mut Statement<'a>, ctx: &mut TraverseCtx<'a>) { - if let Statement::ForOfStatement(for_of) = stmt { - if !for_of.r#await { - return; - } - - // We need to replace the current statement with new statements, - // but we don't have a such method to do it, so we leverage the statement injector. - // - // Now, we use below steps to workaround it: - // 1. Use the last statement as the new statement. - // 2. insert the rest of the statements before the current statement. - // TODO: Once we have a method to replace the current statement, we can simplify this logic. - let mut statements = self.transform_for_of_statement(for_of, ctx); - let last_statement = statements.pop().unwrap(); - *stmt = last_statement; - self.ctx.statement_injector.insert_many_before(&stmt.address(), statements); - } + self.transform_statement(stmt, ctx); } fn exit_statement(&mut self, stmt: &mut Statement<'a>, ctx: &mut TraverseCtx<'a>) { diff --git a/tasks/transform_conformance/snapshots/oxc.snap.md b/tasks/transform_conformance/snapshots/oxc.snap.md index 88c539f886454b..b0bde67a966f57 100644 --- a/tasks/transform_conformance/snapshots/oxc.snap.md +++ b/tasks/transform_conformance/snapshots/oxc.snap.md @@ -1,6 +1,6 @@ commit: d20b314c -Passed: 76/85 +Passed: 77/86 # All Passed: * babel-plugin-transform-class-static-block diff --git a/tasks/transform_conformance/tests/babel-plugin-transform-async-generator-functions/test/fixtures/for-await/with-labeled-statement/input.js b/tasks/transform_conformance/tests/babel-plugin-transform-async-generator-functions/test/fixtures/for-await/with-labeled-statement/input.js new file mode 100644 index 00000000000000..08043c90354497 --- /dev/null +++ b/tasks/transform_conformance/tests/babel-plugin-transform-async-generator-functions/test/fixtures/for-await/with-labeled-statement/input.js @@ -0,0 +1,10 @@ +async function* handleAsyncIterable(asyncIterable) { + outer: for await (const chunk of asyncIterable) { + for (;;) { + if (delimIndex === -1) { + // incomplete message, wait for more chunks + continue outer; + } + } + } +} \ No newline at end of file diff --git a/tasks/transform_conformance/tests/babel-plugin-transform-async-generator-functions/test/fixtures/for-await/with-labeled-statement/output.js b/tasks/transform_conformance/tests/babel-plugin-transform-async-generator-functions/test/fixtures/for-await/with-labeled-statement/output.js new file mode 100644 index 00000000000000..7698295211d563 --- /dev/null +++ b/tasks/transform_conformance/tests/babel-plugin-transform-async-generator-functions/test/fixtures/for-await/with-labeled-statement/output.js @@ -0,0 +1,37 @@ +function handleAsyncIterable(_x) { + return _handleAsyncIterable.apply(this, arguments); +} +function _handleAsyncIterable() { + _handleAsyncIterable = babelHelpers.wrapAsyncGenerator(function* (asyncIterable) { + var _iteratorAbruptCompletion = false; + var _didIteratorError = false; + var _iteratorError; + try { + outer: for (var _iterator = babelHelpers.asyncIterator(asyncIterable), _step; _iteratorAbruptCompletion = !(_step = yield babelHelpers.awaitAsyncGenerator(_iterator.next())).done; _iteratorAbruptCompletion = false) { + const chunk = _step.value; + { + for (;;) { + if (delimIndex === -1) { + // incomplete message, wait for more chunks + continue outer; + } + } + } + } + } catch (err) { + _didIteratorError = true; + _iteratorError = err; + } finally { + try { + if (_iteratorAbruptCompletion && _iterator.return != null) { + yield babelHelpers.awaitAsyncGenerator(_iterator.return()); + } + } finally { + if (_didIteratorError) { + throw _iteratorError; + } + } + } + }); + return _handleAsyncIterable.apply(this, arguments); +} \ No newline at end of file