From 5d7503c6cb920d17bf148b73631a81b51958e2aa Mon Sep 17 00:00:00 2001 From: Rob Hogan Date: Mon, 11 Nov 2024 14:21:52 -0800 Subject: [PATCH] import-export-plugin: Rename potentially conflicting declarations at module scope (module, exports, require, global) (#1385) Summary: Currently, when using `experimentalImportPlugin` (`import-export-plugin`) without `babel-plugin-transform-module-commonjs`, the following completely legal ESM has broken exports and broken internal behaviour: ``` const exports = {} export const foo = 'bar'; process.nextTick(() => { console.log(exports); // should print "{}" }); ``` Because it transforms to this, where `var exports` does not throw (as `const` or `let` would for an already-bound name) but it does shadow the injected `exports` argument, so that no assignment to the injected `exports` is ever made (in other words, a consumer would see no defined exports). In fact, an assignment to the local `exports` *is* made where it shouldn't be. ``` __d(function (global, require, _$$_IMPORT_DEFAULT, _$$_IMPORT_ALL, module, exports, _$$_METRO_DEPENDENCY_MAP) { "use strict"; Object.defineProperty(exports, '__esModule', { value: true }); var exports = {}; var foo = 'bar'; process.nextTick(() => { console.log(exports); // "{foo: 'bar'}" }); exports.foo = foo; },/*...*/); ``` Moreover, in a configuration where we don't transform `const`/`let` to `var` (eg targeting web or server), this actually throws at runtime, as would binding any of `module`, `global` or `require`. The solution is to rename bindings to names that would be fine in ESM but conflict with CommonJS names / Metro's module factory The analogue in the (roughly) corresponding Babel plugin is here: https://github.com/babel/babel/blob/38d26cd5eeb66b697671cfb8c78f963f02992073/packages/babel-plugin-transform-modules-commonjs/src/index.ts#L198-L204 (Note that we do not inject `__dirname` or `__filename` in Metro, but we do inject `global`) Changelog: ``` **[Fix]**: import-exports-plugin: Rename `module`/`require`/`exports`/`global` declarations to avoid conflicts with injected args or generated code. ``` Reviewed By: huntie Differential Revision: D65536715 --- .../__tests__/import-export-plugin-test.js | 73 +++++++++++++++++++ .../src/import-export-plugin.js | 8 ++ 2 files changed, 81 insertions(+) diff --git a/packages/metro-transform-plugins/src/__tests__/import-export-plugin-test.js b/packages/metro-transform-plugins/src/__tests__/import-export-plugin-test.js index 6145ff546d..d0f347b842 100644 --- a/packages/metro-transform-plugins/src/__tests__/import-export-plugin-test.js +++ b/packages/metro-transform-plugins/src/__tests__/import-export-plugin-test.js @@ -186,6 +186,61 @@ test('exports named members', () => { compare([importExportPlugin], code, expected, opts); }); +test('renames existing `exports` declarations in module scope', () => { + const code = ` + const exports = 'foo'; + export const bar = 'bar'; + console.log(exports, bar); + `; + + const expected = ` + Object.defineProperty(exports, '__esModule', {value: true}); + const _exports = 'foo'; + const bar = 'bar'; + console.log(_exports, bar); + exports.bar = bar; + `; + + compare([importExportPlugin], code, expected, opts); +}); + +test('handles an export named "exports"', () => { + const code = ` + export const exports = {a: 'foo'}; + `; + + const expected = ` + Object.defineProperty(exports, '__esModule', {value: true}); + const _exports = { + a: 'foo', + }; + exports.exports = _exports; + `; + + compare([importExportPlugin], code, expected, opts); +}); + +test('allows mixed esm and cjs exports', () => { + const code = ` + export const foo = 'foo'; + exports.bar = 'bar'; + module.exports.baz = 'baz'; + export default class {} + `; + + const expected = ` + Object.defineProperty(exports, '__esModule', {value: true}); + const foo = 'foo'; + exports.bar = 'bar'; + module.exports.baz = 'baz'; + class _default {} + exports.default = _default; + exports.foo = foo; + `; + + compare([importExportPlugin], code, expected, opts); +}); + test('exports destructured named object members', () => { const code = ` export const {foo,bar} = {foo: 'bar',bar: 'baz'}; @@ -266,6 +321,24 @@ test('enables module exporting when something is exported', () => { `); }); +test('renames bindings', () => { + const code = ` + const module = 'foo'; + let exports = 'bar'; + var global = 'baz'; + const require = {}; + `; + + const expected = ` + const _module = 'foo'; + let _exports = 'bar'; + var _global = 'baz'; + const _require = {}; + `; + + compare([importExportPlugin], code, expected, opts); +}); + test('supports `import {default as LocalName}`', () => { const code = ` import { diff --git a/packages/metro-transform-plugins/src/import-export-plugin.js b/packages/metro-transform-plugins/src/import-export-plugin.js index 9cf2f09835..ee3eb63987 100644 --- a/packages/metro-transform-plugins/src/import-export-plugin.js +++ b/packages/metro-transform-plugins/src/import-export-plugin.js @@ -495,6 +495,14 @@ function importExportPlugin({types: t}: {types: Types, ...}): PluginObj { state.imports = []; state.importAll = t.identifier(state.opts.importAll); state.importDefault = t.identifier(state.opts.importDefault); + + // Rename declarations at module scope that might otherwise conflict + // with arguments we inject into the module factory. + // Note that it isn't necessary to rename importAll/importDefault + // because Metro already uses generateUid to generate unused names. + ['module', 'global', 'exports', 'require'].forEach(name => + path.scope.rename(name), + ); }, exit(path: NodePath, state: State): void {