From f7759e6f3aaad25314f1bfad9125b816f234f543 Mon Sep 17 00:00:00 2001 From: Stuart Miller Date: Thu, 9 Jun 2022 17:35:05 +1200 Subject: [PATCH] swc: Support paths from tsconfig The swc transpiler can support paths, but they have to be converted to absolute paths using the baseUrl first. --- src/test/transpilers.spec.ts | 28 ++++++++++++++++++++++++++-- src/transpilers/swc.ts | 16 ++++++++++++++++ tests/swc-with-paths/index.ts | 1 + tests/swc-with-paths/tsconfig.json | 12 ++++++++++++ 4 files changed, 55 insertions(+), 2 deletions(-) create mode 100644 tests/swc-with-paths/index.ts create mode 100644 tests/swc-with-paths/tsconfig.json diff --git a/src/test/transpilers.spec.ts b/src/test/transpilers.spec.ts index 57174f5e8..f3c6bf073 100644 --- a/src/test/transpilers.spec.ts +++ b/src/test/transpilers.spec.ts @@ -3,13 +3,23 @@ // Should consolidate them here. import { context } from './testlib'; -import { ctxTsNode, testsDirRequire } from './helpers'; +import { + CMD_TS_NODE_WITHOUT_PROJECT_FLAG, + ctxTsNode, + TEST_DIR, + testsDirRequire, +} from './helpers'; import * as expect from 'expect'; +import { createExec } from './exec-helpers'; + +const exec = createExec({ + cwd: TEST_DIR, +}); const test = context(ctxTsNode); test.suite('swc', (test) => { - test('verify that TS->SWC target mappings suppport all possible values from both TS and SWC', async (t) => { + test('verify that TS->SWC target mappings support all possible values from both TS and SWC', async (t) => { const swcTranspiler = testsDirRequire( 'ts-node/transpilers/swc-experimental' ) as typeof import('../transpilers/swc'); @@ -44,4 +54,18 @@ test.suite('swc', (test) => { expect([...swcTranspiler.targetMapping.values()]).toContain(target); } }); + + test('verify paths are mapped correctly', async (t) => { + const { err, stdout } = await exec( + `${CMD_TS_NODE_WITHOUT_PROJECT_FLAG} swc-with-paths`, + { + env: { + ...process.env, + NODE_OPTIONS: `${process.env.NODE_OPTIONS || ''}}`, + }, + } + ); + expect(err).toBe(null); + expect(stdout).toMatch('Hello, world!\n'); + }); }); diff --git a/src/transpilers/swc.ts b/src/transpilers/swc.ts index 246b70f40..5eb81c10a 100644 --- a/src/transpilers/swc.ts +++ b/src/transpilers/swc.ts @@ -1,3 +1,4 @@ +import * as path from 'path'; import type * as ts from 'typescript'; import type * as swcWasm from '@swc/wasm'; import type * as swcTypes from '@swc/core'; @@ -60,6 +61,8 @@ export function create(createOptions: SwcTranspilerOptions): Transpiler { strict, alwaysStrict, noImplicitUseStrict, + baseUrl, + paths, } = compilerOptions; const nonTsxOptions = createSwcOptions(false); const tsxOptions = createSwcOptions(true); @@ -110,6 +113,16 @@ export function create(createOptions: SwcTranspilerOptions): Transpiler { noImplicitUseStrict === true ? false : true; + + const absolutePaths = Object.fromEntries( + Object.entries(paths || {}) + .filter(([key, values]) => values.length > 0) + .map(([key, values]) => [ + key, + values.map((value) => path.resolve(baseUrl || './', value)), + ]) + ) as { [from: string]: [string] }; + return { sourceMaps: sourceMap, // isModule: true, @@ -144,6 +157,9 @@ export function create(createOptions: SwcTranspilerOptions): Transpiler { } as swcTypes.ReactConfig, }, keepClassNames, + baseUrl, + paths: + Object.keys(absolutePaths).length > 0 ? absolutePaths : undefined, } as swcTypes.JscConfig, }; } diff --git a/tests/swc-with-paths/index.ts b/tests/swc-with-paths/index.ts new file mode 100644 index 000000000..e1b017be4 --- /dev/null +++ b/tests/swc-with-paths/index.ts @@ -0,0 +1 @@ +import '@parent-dir/hello-world'; diff --git a/tests/swc-with-paths/tsconfig.json b/tests/swc-with-paths/tsconfig.json new file mode 100644 index 000000000..d6dfda977 --- /dev/null +++ b/tests/swc-with-paths/tsconfig.json @@ -0,0 +1,12 @@ +{ + "ts-node": { + "swc": true + }, + "compilerOptions": { + "module": "CommonJS", + "baseUrl": "./", + "paths": { + "@parent-dir/*": ["../*"] + } + } +}