Skip to content

Commit 99e48bb

Browse files
committed
Add path mapping for TypeScript and node imports
Closes tapjs#11 tapjs#12
1 parent 2d9f4ea commit 99e48bb

File tree

7 files changed

+156
-25
lines changed

7 files changed

+156
-25
lines changed

package-lock.json

+2-2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

package.json

+2
Original file line numberDiff line numberDiff line change
@@ -57,8 +57,10 @@
5757
"@isaacs/cached": "^1.0.1",
5858
"@isaacs/catcher": "^1.0.4",
5959
"foreground-child": "^3.1.1",
60+
"get-tsconfig": "^4.7.2",
6061
"mkdirp": "^3.0.1",
6162
"pirates": "^4.0.6",
63+
"resolve-pkg-maps": "^1.0.0",
6264
"rimraf": "^6.0.1",
6365
"signal-exit": "^4.1.0",
6466
"sock-daemon": "^1.4.2",

src/hooks/hooks.mts

+2
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@ import { MessagePort } from 'node:worker_threads'
1313
import { classifyModule } from '../classify-module.js'
1414
import { DaemonClient } from '../client.js'
1515
import { getDiagMode } from '../diagnostic-mode.js'
16+
import { resolveMapping } from '../service/resolve-mapping.js'
1617

1718
// in some cases on the loader thread, console.error doesn't actually
1819
// print. sync write to fd 1 instead.
@@ -54,6 +55,7 @@ export const resolve: ResolveHook = async (
5455
context,
5556
nextResolve,
5657
) => {
58+
url = resolveMapping(url, context.parentURL)
5759
const { parentURL } = context
5860
const target =
5961
/* c8 ignore start */

src/service/resolve-mapping.ts

+93
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,93 @@
1+
import { tsconfig } from './tsconfig.js'
2+
import { relative, dirname, join } from 'node:path'
3+
import { fileURLToPath } from 'node:url'
4+
import { createPathsMatcher, type TsConfigResult } from 'get-tsconfig'
5+
import {
6+
resolveImports,
7+
type PathConditionsMap,
8+
} from 'resolve-pkg-maps'
9+
import { walkUp } from 'walk-up-path'
10+
import { catcher } from '@isaacs/catcher'
11+
import { readFile } from '../ts-sys-cached.js'
12+
import { classifyModule } from '../classify-module.js'
13+
14+
const resolveTypescriptMapping = (
15+
url: string,
16+
parent: string | undefined
17+
): string | undefined => {
18+
if (!url || url.startsWith('file:///') || !parent) return undefined
19+
const options = tsconfig().options
20+
21+
const relativePath = url
22+
const tsconfigArg = {
23+
config: { compilerOptions: { ...options, baseUrl: './' } },
24+
path: options.configFilePath as string,
25+
} as TsConfigResult
26+
27+
const pathMather = createPathsMatcher(tsconfigArg)
28+
29+
if (!pathMather) return undefined
30+
const found = pathMather(relativePath)
31+
if (found.length === 0) return undefined
32+
return relative(
33+
dirname(fileURLToPath(parent)),
34+
found[0] as string
35+
).toString()
36+
}
37+
38+
const getPackageJson = (
39+
from: string
40+
):
41+
| ({ imports: PathConditionsMap } & { path: string })
42+
| undefined => {
43+
for (const d of walkUp(from)) {
44+
const pj = catcher(() => {
45+
const json = readFile(d + '/package.json')
46+
if (!json) return undefined
47+
const pj = JSON.parse(json) as { imports: PathConditionsMap }
48+
return pj
49+
})
50+
if (pj) {
51+
return { imports: pj?.imports, path: d }
52+
}
53+
}
54+
return undefined
55+
}
56+
57+
const resolvePackageJsonMapping = (
58+
url: string,
59+
parent: string | undefined
60+
): string | undefined => {
61+
if (
62+
!url ||
63+
url.startsWith('file:///') ||
64+
!url.startsWith('#') ||
65+
!parent
66+
)
67+
return undefined
68+
69+
const options = getPackageJson(dirname(fileURLToPath(parent)))
70+
if (options === undefined) return undefined
71+
72+
const found = resolveImports(options.imports, url, [
73+
'default',
74+
'node',
75+
classifyModule(url) === 'commonjs' ? 'require' : 'import',
76+
])
77+
if (found.length === 0) return undefined
78+
return relative(
79+
dirname(fileURLToPath(parent)),
80+
join(options.path, found[0] as string)
81+
).toString()
82+
}
83+
84+
export const resolveMapping = (
85+
url: string,
86+
parent: string | undefined
87+
): string => {
88+
return (
89+
resolvePackageJsonMapping(url, parent) ??
90+
resolveTypescriptMapping(url, parent) ??
91+
url
92+
)
93+
}

src/service/transpile-only.ts

+1
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,7 @@ const createTsTranspileModule = ({
6666
) {
6767
continue
6868
}
69+
if (option.name === 'paths') continue
6970

7071
options[option.name] = option.transpileOptionValue
7172
}

src/service/tsconfig.ts

+7-3
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55
// return the same object if it parses to the same values.
66
import { catcher } from '@isaacs/catcher'
77
import { statSync } from 'fs'
8-
import { resolve } from 'path'
8+
import { dirname, resolve } from 'node:path'
99
import ts from 'typescript'
1010
import { walkUp } from 'walk-up-path'
1111
import { error, warn } from '../debug.js'
@@ -73,7 +73,7 @@ export const tsconfig = () => {
7373
// also default to recommended setting for node programs
7474
{
7575
compilerOptions: {
76-
rootDir: dir,
76+
rootDir: dirname(configPath),
7777
skipLibCheck: true,
7878
isolatedModules: true,
7979
esModuleInterop: true,
@@ -106,7 +106,11 @@ export const tsconfig = () => {
106106
noEmit: false,
107107
},
108108
})
109-
const newConfig = ts.parseJsonConfigFileContent(res, ts.sys, dir)
109+
const newConfig = ts.parseJsonConfigFileContent(
110+
res,
111+
ts.sys,
112+
dirname(configPath)
113+
)
110114
const newConfigJSON = JSON.stringify(newConfig)
111115
if (loadedConfig && newConfigJSON === loadedConfigJSON) {
112116
// no changes, keep the old one

test/bin.ts

+49-20
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,13 @@ const bin = fileURLToPath(
88
new URL('../dist/esm/bin.mjs', import.meta.url),
99
)
1010

11-
const run = (args: string[]) =>
12-
spawnSync(process.execPath, [bin, ...args], { encoding: 'utf8' })
11+
const run = (args: string[], tsconfigPath?: string) =>
12+
spawnSync(process.execPath, [bin, ...args], {
13+
encoding: 'utf8',
14+
env: {
15+
TSIMP_PROJECT: tsconfigPath,
16+
},
17+
})
1318

1419
t.teardown(() => run(['--stop']))
1520

@@ -68,30 +73,33 @@ t.test('actually run a program', async t => {
6873

6974
// subpath import tests
7075
'package.json': JSON.stringify({
71-
"type": "module",
72-
"imports": {
73-
"#utilities/*": "./utilities/*"
74-
}
76+
type: 'module',
77+
imports: {
78+
'#utilities/*': './utilities/*',
79+
},
7580
}),
7681
'tsconfig.json': JSON.stringify({
77-
"compilerOptions": {
78-
"baseUrl": ".",
79-
"paths": {
80-
"#utilities/*": ["./utilities/*"],
81-
}
82+
compilerOptions: {
83+
baseUrl: '.',
84+
resolvePackageJsonImports: true,
85+
paths: {
86+
'+utilities/*': ['./utilities/*'],
87+
},
8288
},
8389
}),
8490

85-
'utilities': {
86-
'source': {
87-
'constants.ts': 'enum Constants { one = "one", two = "two" }; export default Constants;'
88-
}
91+
utilities: {
92+
source: {
93+
'constants.ts':
94+
'enum Constants { one = "one", two = "two", three = "three" }; export default Constants;',
95+
},
8996
},
9097

91-
"test": {
98+
test: {
9299
'getOne.ts': `import Constants from "../utilities/source/constants.js"; console.log(Constants.one);`,
93-
'getTwo.ts': `import Constants from "#utilities/source/constants.js"; console.log(Constants.two);`
94-
}
100+
'getTwo.ts': `import Constants from "#utilities/source/constants.ts"; console.log(Constants.two);`,
101+
'getThree.ts': `import Constants from "+utilities/source/constants.js"; console.log(Constants.three);`,
102+
},
95103
})
96104
const rel = relative(process.cwd(), dir).replace(/\\/g, '/')
97105

@@ -154,18 +162,39 @@ t.test('actually run a program', async t => {
154162
})
155163

156164
t.test('run file with subpath imports', async t => {
165+
run(['--restart'], `./${rel}/tsconfig.json`)
166+
157167
{
158168
const pathToFile = `./${rel}/test/getOne.ts`
159-
const { stdout, status } = run([pathToFile])
169+
const { stdout, status } = run(
170+
[pathToFile],
171+
`./${rel}/tsconfig.json`
172+
)
173+
160174
t.equal(status, 0)
161175
t.equal(stdout, 'one\n')
162176
}
163177

164178
{
165179
const pathToFile = `./${rel}/test/getTwo.ts`
166-
const { stdout, status } = run([pathToFile])
180+
const { stdout, status } = run(
181+
[pathToFile],
182+
`./${rel}/tsconfig.json`
183+
)
184+
167185
t.equal(status, 0)
168186
t.equal(stdout, 'two\n')
169187
}
188+
189+
{
190+
const pathToFile = `./${rel}/test/getThree.ts`
191+
const { stdout, status } = run(
192+
[pathToFile],
193+
`./${rel}/tsconfig.json`
194+
)
195+
196+
t.equal(status, 0)
197+
t.equal(stdout, 'three\n')
198+
}
170199
})
171200
})

0 commit comments

Comments
 (0)