From b4cbe8e4790d65e8080652cd18bfec078ca7ecab Mon Sep 17 00:00:00 2001 From: Janka Uryga Date: Fri, 20 Dec 2024 00:23:06 +0100 Subject: [PATCH 1/2] fix(turbopack): use explicit mjs/cjs extension for virtual proxy file --- .../ecmascript_client_reference_proxy_module.rs | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/crates/next-core/src/next_client_reference/ecmascript_client_reference/ecmascript_client_reference_proxy_module.rs b/crates/next-core/src/next_client_reference/ecmascript_client_reference/ecmascript_client_reference_proxy_module.rs index 0965990070f27..3cabfee3b2550 100644 --- a/crates/next-core/src/next_client_reference/ecmascript_client_reference/ecmascript_client_reference_proxy_module.rs +++ b/crates/next-core/src/next_client_reference/ecmascript_client_reference/ecmascript_client_reference_proxy_module.rs @@ -70,11 +70,13 @@ impl EcmascriptClientReferenceProxyModule { #[turbo_tasks::function] async fn proxy_module(&self) -> Result>> { let mut code = CodeBuilder::default(); + let is_esm: bool; let server_module_path = &*self.server_module_ident.to_string().await?; // Adapted from https://github.com/facebook/react/blob/c5b9375767e2c4102d7e5559d383523736f1c902/packages/react-server-dom-webpack/src/ReactFlightWebpackNodeLoader.js#L323-L354 if let EcmascriptExports::EsmExports(exports) = &*self.client_module.get_exports().await? { + is_esm = true; let exports = exports.expand_exports().await?; if !exports.dynamic_exports.is_empty() { @@ -130,6 +132,7 @@ impl EcmascriptClientReferenceProxyModule { } } } else { + is_esm = false; writedoc!( code, r#" @@ -146,7 +149,13 @@ impl EcmascriptClientReferenceProxyModule { AssetContent::file(File::from(code.source_code().clone()).into()); let proxy_source = VirtualSource::new( - self.server_module_ident.path().join("proxy.js".into()), + self.server_module_ident.path().join( + // Depending on the original format, we call the file `proxy.mjs` or `proxy.cjs`. + // This is because we're placing the virtual module next to the original code, so + // its parsing will be affected by `type` fields in package.json -- + // a bare `proxy.js` may end up being unexpectedly parsed as the wrong format. + format!("proxy.{}", if is_esm { "mjs" } else { "cjs" }).into(), + ), proxy_module_content, ); From 520d4e1e84c3f0adeab645865afe2c84f4651169 Mon Sep 17 00:00:00 2001 From: Janka Uryga Date: Fri, 20 Dec 2024 01:56:39 +0100 Subject: [PATCH 2/2] test: add tests --- .../app/import-cjs/page.tsx | 11 +++++++ .../app/import-esm/page.tsx | 11 +++++++ .../app/layout.tsx | 8 +++++ .../app/require-cjs/page.tsx | 11 +++++++ .../app/require-esm/page.tsx | 11 +++++++ .../index.test.ts | 31 +++++++++++++++++++ .../next.config.js | 6 ++++ .../node_modules/lib-cjs/index.js | 3 ++ .../node_modules/lib-cjs/index.mjs | 3 ++ .../node_modules/lib-cjs/package.json | 10 ++++++ .../node_modules/lib-esm/index.cjs | 3 ++ .../node_modules/lib-esm/index.js | 3 ++ .../node_modules/lib-esm/package.json | 10 ++++++ 13 files changed, 121 insertions(+) create mode 100644 test/e2e/app-dir/client-module-with-package-type/app/import-cjs/page.tsx create mode 100644 test/e2e/app-dir/client-module-with-package-type/app/import-esm/page.tsx create mode 100644 test/e2e/app-dir/client-module-with-package-type/app/layout.tsx create mode 100644 test/e2e/app-dir/client-module-with-package-type/app/require-cjs/page.tsx create mode 100644 test/e2e/app-dir/client-module-with-package-type/app/require-esm/page.tsx create mode 100644 test/e2e/app-dir/client-module-with-package-type/index.test.ts create mode 100644 test/e2e/app-dir/client-module-with-package-type/next.config.js create mode 100644 test/e2e/app-dir/client-module-with-package-type/node_modules/lib-cjs/index.js create mode 100644 test/e2e/app-dir/client-module-with-package-type/node_modules/lib-cjs/index.mjs create mode 100644 test/e2e/app-dir/client-module-with-package-type/node_modules/lib-cjs/package.json create mode 100644 test/e2e/app-dir/client-module-with-package-type/node_modules/lib-esm/index.cjs create mode 100644 test/e2e/app-dir/client-module-with-package-type/node_modules/lib-esm/index.js create mode 100644 test/e2e/app-dir/client-module-with-package-type/node_modules/lib-esm/package.json diff --git a/test/e2e/app-dir/client-module-with-package-type/app/import-cjs/page.tsx b/test/e2e/app-dir/client-module-with-package-type/app/import-cjs/page.tsx new file mode 100644 index 0000000000000..60befa195889b --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/app/import-cjs/page.tsx @@ -0,0 +1,11 @@ +import * as React from 'react' + +import EsmFromCjs from 'lib-cjs' + +export default function Page() { + return ( +

+ lib-cjs: +

+ ) +} diff --git a/test/e2e/app-dir/client-module-with-package-type/app/import-esm/page.tsx b/test/e2e/app-dir/client-module-with-package-type/app/import-esm/page.tsx new file mode 100644 index 0000000000000..4469d96d5dc26 --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/app/import-esm/page.tsx @@ -0,0 +1,11 @@ +import * as React from 'react' + +import EsmFromEsm from 'lib-esm' + +export default function Page() { + return ( +

+ lib-esm: +

+ ) +} diff --git a/test/e2e/app-dir/client-module-with-package-type/app/layout.tsx b/test/e2e/app-dir/client-module-with-package-type/app/layout.tsx new file mode 100644 index 0000000000000..888614deda3ba --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/app/layout.tsx @@ -0,0 +1,8 @@ +import { ReactNode } from 'react' +export default function Root({ children }: { children: ReactNode }) { + return ( + + {children} + + ) +} diff --git a/test/e2e/app-dir/client-module-with-package-type/app/require-cjs/page.tsx b/test/e2e/app-dir/client-module-with-package-type/app/require-cjs/page.tsx new file mode 100644 index 0000000000000..1dc7a4bf66786 --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/app/require-cjs/page.tsx @@ -0,0 +1,11 @@ +import * as React from 'react' + +const CjsFromCjs = require('lib-cjs') + +export default function Page() { + return ( +

+ lib-cjs: +

+ ) +} diff --git a/test/e2e/app-dir/client-module-with-package-type/app/require-esm/page.tsx b/test/e2e/app-dir/client-module-with-package-type/app/require-esm/page.tsx new file mode 100644 index 0000000000000..fb20339de17b0 --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/app/require-esm/page.tsx @@ -0,0 +1,11 @@ +import * as React from 'react' + +const CjsFromEsm = require('lib-esm') + +export default function Page() { + return ( +

+ lib-esm: +

+ ) +} diff --git a/test/e2e/app-dir/client-module-with-package-type/index.test.ts b/test/e2e/app-dir/client-module-with-package-type/index.test.ts new file mode 100644 index 0000000000000..7b70d442bab65 --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/index.test.ts @@ -0,0 +1,31 @@ +import { nextTestSetup } from 'e2e-utils' + +describe('esm-client-module-without-exports', () => { + const { next } = nextTestSetup({ + files: __dirname, + }) + + describe('"type": "commonjs" in package.json', () => { + it('should render without errors: import cjs', async () => { + const $ = await next.render$('/import-cjs') + expect($('p').text()).toContain('lib-cjs: esm') + }) + + it('should render without errors: require cjs', async () => { + const $ = await next.render$('/require-cjs') + expect($('p').text()).toContain('lib-cjs: cjs') + }) + }) + + describe('"type": "module" in package.json', () => { + it('should render without errors: import esm', async () => { + const $ = await next.render$('/import-esm') + expect($('p').text()).toContain('lib-esm: esm') + }) + + it('should render without errors: require esm', async () => { + const $ = await next.render$('/require-esm') + expect($('p').text()).toContain('lib-esm: cjs') + }) + }) +}) diff --git a/test/e2e/app-dir/client-module-with-package-type/next.config.js b/test/e2e/app-dir/client-module-with-package-type/next.config.js new file mode 100644 index 0000000000000..807126e4cf0bf --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/next.config.js @@ -0,0 +1,6 @@ +/** + * @type {import('next').NextConfig} + */ +const nextConfig = {} + +module.exports = nextConfig diff --git a/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-cjs/index.js b/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-cjs/index.js new file mode 100644 index 0000000000000..271c41e8ce833 --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-cjs/index.js @@ -0,0 +1,3 @@ +'use client' +console.log('lib-cjs :: cjs') +module.exports = () => 'cjs' diff --git a/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-cjs/index.mjs b/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-cjs/index.mjs new file mode 100644 index 0000000000000..8a112198a72ef --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-cjs/index.mjs @@ -0,0 +1,3 @@ +'use client' +console.log('lib-cjs :: esm') +export default () => 'esm' diff --git a/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-cjs/package.json b/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-cjs/package.json new file mode 100644 index 0000000000000..a8d5b97fc0b92 --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-cjs/package.json @@ -0,0 +1,10 @@ +{ + "name": "lib-cjs", + "type": "commonjs", + "exports": { + ".": { + "import": "./index.mjs", + "default": "./index.js" + } + } +} diff --git a/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-esm/index.cjs b/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-esm/index.cjs new file mode 100644 index 0000000000000..0b2a17179b5d3 --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-esm/index.cjs @@ -0,0 +1,3 @@ +'use client' +console.log('lib-esm :: cjs') +module.exports = () => 'cjs' diff --git a/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-esm/index.js b/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-esm/index.js new file mode 100644 index 0000000000000..2fc1856c0840a --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-esm/index.js @@ -0,0 +1,3 @@ +'use client' +console.log('lib-esm :: esm') +export default () => 'esm' diff --git a/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-esm/package.json b/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-esm/package.json new file mode 100644 index 0000000000000..e432daa17225d --- /dev/null +++ b/test/e2e/app-dir/client-module-with-package-type/node_modules/lib-esm/package.json @@ -0,0 +1,10 @@ +{ + "name": "lib-esm", + "type": "module", + "exports": { + ".": { + "require": "./index.cjs", + "default": "./index.js" + } + } +}