From ed818d9cc6560e32d90832296a289caf616dded7 Mon Sep 17 00:00:00 2001 From: overlookmotel Date: Fri, 6 Dec 2024 19:41:11 +0000 Subject: [PATCH] fix(transformer/class-properties): make `_super` function outside class strict mode --- .../es2022/class_properties/constructor.rs | 17 +++++++++++++-- .../output.js | 1 + .../output.js | 1 + .../derived-super-in-default-params/output.js | 1 + .../snapshots/babel.snap.md | 9 -------- .../snapshots/oxc.snap.md | 13 ++++++++++-- .../super-in-constructor-strict/input.js | 14 +++++++++++++ .../super-in-constructor-strict/output.js | 21 +++++++++++++++++++ 8 files changed, 64 insertions(+), 13 deletions(-) create mode 100644 tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/super-in-constructor-strict/input.js create mode 100644 tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/super-in-constructor-strict/output.js diff --git a/crates/oxc_transformer/src/es2022/class_properties/constructor.rs b/crates/oxc_transformer/src/es2022/class_properties/constructor.rs index 1b7dfafc0641a3..15eabe38534f68 100644 --- a/crates/oxc_transformer/src/es2022/class_properties/constructor.rs +++ b/crates/oxc_transformer/src/es2022/class_properties/constructor.rs @@ -87,6 +87,7 @@ //! Oxc output: //! ```js //! let _super = function() { +//! "use strict"; //! this.prop = foo(); //! return this; //! }; @@ -407,8 +408,20 @@ impl<'a, 'c> ConstructorParamsSuperReplacer<'a, 'c> { let super_func_scope_id = ctx.scopes_mut().add_scope( Some(outer_scope_id), NodeId::DUMMY, - ScopeFlags::Function | ScopeFlags::Arrow | ScopeFlags::StrictMode, + ScopeFlags::Function | ScopeFlags::StrictMode, ); + + // Add `"use strict"` directive if outer scope is not strict mode + let directives = if ctx.scopes().get_flags(outer_scope_id).is_strict_mode() { + ctx.ast.vec() + } else { + ctx.ast.vec1(ctx.ast.directive( + SPAN, + ctx.ast.string_literal(SPAN, Atom::from("use strict"), None), + Atom::from("use strict"), + )) + }; + // `return this;` let return_stmt = ctx.ast.statement_return(SPAN, Some(ctx.ast.expression_this(SPAN))); // `; return this;` @@ -430,7 +443,7 @@ impl<'a, 'c> ConstructorParamsSuperReplacer<'a, 'c> { NONE, ), NONE, - Some(ctx.ast.alloc_function_body(SPAN, ctx.ast.vec(), body_stmts)), + Some(ctx.ast.alloc_function_body(SPAN, directives, body_stmts)), super_func_scope_id, )) } diff --git a/tasks/transform_conformance/overrides/babel-plugin-transform-class-properties/test/fixtures/public/derived-super-in-default-params-complex/output.js b/tasks/transform_conformance/overrides/babel-plugin-transform-class-properties/test/fixtures/public/derived-super-in-default-params-complex/output.js index 9c789497f69fd3..81b3631788f122 100644 --- a/tasks/transform_conformance/overrides/babel-plugin-transform-class-properties/test/fixtures/public/derived-super-in-default-params-complex/output.js +++ b/tasks/transform_conformance/overrides/babel-plugin-transform-class-properties/test/fixtures/public/derived-super-in-default-params-complex/output.js @@ -1,4 +1,5 @@ let _super = function() { + "use strict"; babelHelpers.defineProperty(this, "bar", "foo"); return this; }; diff --git a/tasks/transform_conformance/overrides/babel-plugin-transform-class-properties/test/fixtures/public/derived-super-in-default-params-in-arrow/output.js b/tasks/transform_conformance/overrides/babel-plugin-transform-class-properties/test/fixtures/public/derived-super-in-default-params-in-arrow/output.js index 48b8e56378f94f..5a4f1154f4463d 100644 --- a/tasks/transform_conformance/overrides/babel-plugin-transform-class-properties/test/fixtures/public/derived-super-in-default-params-in-arrow/output.js +++ b/tasks/transform_conformance/overrides/babel-plugin-transform-class-properties/test/fixtures/public/derived-super-in-default-params-in-arrow/output.js @@ -1,4 +1,5 @@ let _super = function() { + "use strict"; babelHelpers.defineProperty(this, "bar", "foo"); return this; }; diff --git a/tasks/transform_conformance/overrides/babel-plugin-transform-class-properties/test/fixtures/public/derived-super-in-default-params/output.js b/tasks/transform_conformance/overrides/babel-plugin-transform-class-properties/test/fixtures/public/derived-super-in-default-params/output.js index 985fbb012ab074..a78c74e145944b 100644 --- a/tasks/transform_conformance/overrides/babel-plugin-transform-class-properties/test/fixtures/public/derived-super-in-default-params/output.js +++ b/tasks/transform_conformance/overrides/babel-plugin-transform-class-properties/test/fixtures/public/derived-super-in-default-params/output.js @@ -1,4 +1,5 @@ let _super = function() { + "use strict"; babelHelpers.defineProperty(this, "bar", "foo"); return this; }; diff --git a/tasks/transform_conformance/snapshots/babel.snap.md b/tasks/transform_conformance/snapshots/babel.snap.md index 73697f9a71a303..d032130e08c84c 100644 --- a/tasks/transform_conformance/snapshots/babel.snap.md +++ b/tasks/transform_conformance/snapshots/babel.snap.md @@ -822,25 +822,16 @@ x Output mismatch x Output mismatch * public/derived-super-in-default-params/input.js -Scope flags mismatch: -after transform: ScopeId(3): ScopeFlags(StrictMode | Function | Arrow) -rebuilt : ScopeId(1): ScopeFlags(Function) Symbol flags mismatch for "_super": after transform: SymbolId(2): SymbolFlags(FunctionScopedVariable) rebuilt : SymbolId(0): SymbolFlags(BlockScopedVariable) * public/derived-super-in-default-params-complex/input.js -Scope flags mismatch: -after transform: ScopeId(3): ScopeFlags(StrictMode | Function | Arrow) -rebuilt : ScopeId(1): ScopeFlags(Function) Symbol flags mismatch for "_super": after transform: SymbolId(2): SymbolFlags(FunctionScopedVariable) rebuilt : SymbolId(0): SymbolFlags(BlockScopedVariable) * public/derived-super-in-default-params-in-arrow/input.js -Scope flags mismatch: -after transform: ScopeId(4): ScopeFlags(StrictMode | Function | Arrow) -rebuilt : ScopeId(1): ScopeFlags(Function) Symbol flags mismatch for "_super": after transform: SymbolId(2): SymbolFlags(FunctionScopedVariable) rebuilt : SymbolId(0): SymbolFlags(BlockScopedVariable) diff --git a/tasks/transform_conformance/snapshots/oxc.snap.md b/tasks/transform_conformance/snapshots/oxc.snap.md index 28790404a475cd..057c5cafb8614a 100644 --- a/tasks/transform_conformance/snapshots/oxc.snap.md +++ b/tasks/transform_conformance/snapshots/oxc.snap.md @@ -1,9 +1,8 @@ commit: 54a8389f -Passed: 94/105 +Passed: 94/106 # All Passed: -* babel-plugin-transform-class-properties * babel-plugin-transform-class-static-block * babel-plugin-transform-nullish-coalescing-operator * babel-plugin-transform-optional-catch-binding @@ -16,6 +15,16 @@ Passed: 94/105 * regexp +# babel-plugin-transform-class-properties (1/2) +* super-in-constructor-strict/input.js +Symbol flags mismatch for "_super": +after transform: SymbolId(6): SymbolFlags(FunctionScopedVariable) +rebuilt : SymbolId(1): SymbolFlags(BlockScopedVariable) +Symbol flags mismatch for "_super2": +after transform: SymbolId(7): SymbolFlags(FunctionScopedVariable) +rebuilt : SymbolId(5): SymbolFlags(BlockScopedVariable) + + # babel-plugin-transform-async-to-generator (14/15) * super/nested/input.js x Output mismatch diff --git a/tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/super-in-constructor-strict/input.js b/tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/super-in-constructor-strict/input.js new file mode 100644 index 00000000000000..c10afb85cbdc49 --- /dev/null +++ b/tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/super-in-constructor-strict/input.js @@ -0,0 +1,14 @@ +function sloppy() { + class C extends S { + prop = 1; + constructor(x = super()) {} + } +} + +function strict() { + "use strict"; + class C extends S { + prop = 1; + constructor(x = super()) {} + } +} diff --git a/tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/super-in-constructor-strict/output.js b/tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/super-in-constructor-strict/output.js new file mode 100644 index 00000000000000..0788916eddfe9b --- /dev/null +++ b/tasks/transform_conformance/tests/babel-plugin-transform-class-properties/test/fixtures/super-in-constructor-strict/output.js @@ -0,0 +1,21 @@ +function sloppy() { + let _super = function() { + "use strict"; + babelHelpers.defineProperty(this, "prop", 1); + return this; + }; + class C extends S { + constructor(x = _super.call(super())) {} + } +} + +function strict() { + "use strict"; + let _super2 = function() { + babelHelpers.defineProperty(this, "prop", 1); + return this; + }; + class C extends S { + constructor(x = _super2.call(super())) {} + } +}