From e2c6db5226e9b34442ab65e02d5bbc3fa9aca65c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Donny/=EA=B0=95=EB=8F=99=EC=9C=A4?= Date: Mon, 18 Mar 2024 14:38:31 +0900 Subject: [PATCH] fix(es/module): Fix interop of `jsc.paths` with symlinks (#8757) **Related issue:** - Closes #8667 --- .../2-cjs/input/src/command/restore.ts | 2 +- .../2-cjs/output/src/command/restore.ts | 2 +- .../2-esm/input/src/command/restore.ts | 2 +- .../2-esm/output/src/command/restore.ts | 2 +- crates/swc_cli_impl/tests/issues.rs | 65 ++++++++++++++++++- crates/swc_ecma_transforms_module/src/path.rs | 6 +- 6 files changed, 71 insertions(+), 8 deletions(-) diff --git a/crates/swc/tests/fixture/deno/paths/issue-2753/2-cjs/input/src/command/restore.ts b/crates/swc/tests/fixture/deno/paths/issue-2753/2-cjs/input/src/command/restore.ts index 8e2af4f0ca71..c45b7ddd4adc 100644 --- a/crates/swc/tests/fixture/deno/paths/issue-2753/2-cjs/input/src/command/restore.ts +++ b/crates/swc/tests/fixture/deno/paths/issue-2753/2-cjs/input/src/command/restore.ts @@ -1 +1 @@ -import { module } from "@src/config"; +import { module } from "@src/conf"; diff --git a/crates/swc/tests/fixture/deno/paths/issue-2753/2-cjs/output/src/command/restore.ts b/crates/swc/tests/fixture/deno/paths/issue-2753/2-cjs/output/src/command/restore.ts index 7793058058c0..dc435deacdc0 100644 --- a/crates/swc/tests/fixture/deno/paths/issue-2753/2-cjs/output/src/command/restore.ts +++ b/crates/swc/tests/fixture/deno/paths/issue-2753/2-cjs/output/src/command/restore.ts @@ -2,4 +2,4 @@ Object.defineProperty(exports, "__esModule", { value: true }); -const _config = require("../config"); +const _conf = require("../conf"); diff --git a/crates/swc/tests/fixture/deno/paths/issue-2753/2-esm/input/src/command/restore.ts b/crates/swc/tests/fixture/deno/paths/issue-2753/2-esm/input/src/command/restore.ts index 8e2af4f0ca71..c45b7ddd4adc 100644 --- a/crates/swc/tests/fixture/deno/paths/issue-2753/2-esm/input/src/command/restore.ts +++ b/crates/swc/tests/fixture/deno/paths/issue-2753/2-esm/input/src/command/restore.ts @@ -1 +1 @@ -import { module } from "@src/config"; +import { module } from "@src/conf"; diff --git a/crates/swc/tests/fixture/deno/paths/issue-2753/2-esm/output/src/command/restore.ts b/crates/swc/tests/fixture/deno/paths/issue-2753/2-esm/output/src/command/restore.ts index cc2bb795da7a..adb86c022798 100644 --- a/crates/swc/tests/fixture/deno/paths/issue-2753/2-esm/output/src/command/restore.ts +++ b/crates/swc/tests/fixture/deno/paths/issue-2753/2-esm/output/src/command/restore.ts @@ -1 +1 @@ -import { module } from "../config"; +import { module } from "../conf"; diff --git a/crates/swc_cli_impl/tests/issues.rs b/crates/swc_cli_impl/tests/issues.rs index 29b7d4220221..5d29cd012456 100644 --- a/crates/swc_cli_impl/tests/issues.rs +++ b/crates/swc_cli_impl/tests/issues.rs @@ -1,5 +1,5 @@ use std::{ - fs::{self, create_dir_all}, + fs::{self, create_dir_all, hard_link}, path::Path, process::{Command, Stdio}, }; @@ -81,6 +81,69 @@ fn issue_8495_1() -> Result<()> { Ok(()) } +#[test] +fn issue_8667_1() -> Result<()> { + let output_base = TempDir::new()?; + let bin_dir = output_base.path().join("bazel-out/arch/bin"); + let sandbox = output_base.path().join("sandbox/123"); + + let pwd = Path::new("tests/fixture-manual/8265").canonicalize()?; + + create_dir_all(bin_dir.join("src/modules/moduleA"))?; + create_dir_all(bin_dir.join("src/modules/moduleB"))?; + create_dir_all(sandbox.join("src/modules/moduleA"))?; + create_dir_all(sandbox.join("src/modules/moduleB"))?; + + // hard links from BINDIR into src + hard_link(pwd.join(".swcrc"), bin_dir.join(".swcrc"))?; + hard_link(pwd.join("src/index.ts"), bin_dir.join("src/index.ts"))?; + hard_link( + pwd.join("src/modules/moduleA/index.ts"), + bin_dir.join("src/modules/moduleA/index.ts"), + )?; + hard_link( + pwd.join("src/modules/moduleB/index.ts"), + bin_dir.join("src/modules/moduleB/index.ts"), + )?; + + // soft links from sandbox into bazel-bin + symlink(&bin_dir.join(".swcrc"), &sandbox.join(".swcrc")); + symlink(&bin_dir.join("src/index.ts"), &sandbox.join("src/index.ts")); + symlink( + &bin_dir.join("src/modules/moduleA/index.ts"), + &sandbox.join("src/modules/moduleA/index.ts"), + ); + symlink( + &bin_dir.join("src/modules/moduleB/index.ts"), + &sandbox.join("src/modules/moduleB/index.ts"), + ); + + // + print_ls_alr(&sandbox); + + let mut cmd = cli()?; + cmd.current_dir(&sandbox) + .arg("compile") + .arg("--source-maps") + .arg("false") + .arg("--config-file") + .arg(".swcrc") + .arg("--out-file") + .arg(bin_dir.join("src/index.js")) + .arg("src/index.ts"); + + cmd.assert().success(); + + let content = fs::read_to_string(bin_dir.join("src/index.js"))?; + assert!( + content.contains("require(\"./modules/moduleA\")"), + "{}", + content + ); + + Ok(()) +} + /// ln -s $a $b fn symlink(a: &Path, b: &Path) { #[cfg(unix)] diff --git a/crates/swc_ecma_transforms_module/src/path.rs b/crates/swc_ecma_transforms_module/src/path.rs index 057f09f73031..06ee744f46a7 100644 --- a/crates/swc_ecma_transforms_module/src/path.rs +++ b/crates/swc_ecma_transforms_module/src/path.rs @@ -1,8 +1,8 @@ use std::{ borrow::Cow, env::current_dir, - fs::read_link, - io::{self}, + fs::canonicalize, + io, path::{Component, Path, PathBuf}, sync::Arc, }; @@ -245,7 +245,7 @@ where // // https://github.com/swc-project/swc/issues/8265 if let FileName::Real(resolved) = &target.filename { - if let Ok(orig) = read_link(resolved) { + if let Ok(orig) = canonicalize(resolved) { target.filename = FileName::Real(orig); } }