无法找到 Typescript 模块,即使 tsc 成功地解决了它
Cannot find Typescript module even though tsc successfully manages to resolve it
我有一个用 Typescript 编写的 Node.js 项目,预计 运行 作为 CLI,但我无法使用 node_modules
目录导入模块绝对路径(相对路径工作正常)。值得一提的是,我正在使用 oclif framework 构建我的 CLI。
我的项目组织如下:
cli
|--node_modules
|--src
|--my-module.ts
|--subdir
|--index.ts
在 my-module.ts
我有:
export class MyClass {
myClassFcn(s: string) {
return 'result'
}
}
index.ts
脚本包含如下内容:
import {MyClass} = require('my-module')
当我尝试使用 ts-node 执行我的应用程序时,我得到
(node:10423) [MODULE_NOT_FOUND] Error Plugin: cli: Cannot find module 'my-module'
module: @oclif/config@1.6.17
task: toCached
plugin: cli
root: /home/eschmidt/Workspace/cli
Error Plugin: cli: Cannot find module 'my-module'
at Function.Module._resolveFilename (internal/modules/cjs/loader.js:571:15)
at Function.Module._load (internal/modules/cjs/loader.js:497:25)
at Module.require (internal/modules/cjs/loader.js:626:17)
at require (internal/modules/cjs/helpers.js:20:18)
at Object.<anonymous> (/home/eschmidt/Workspace/cli/src/commands/create/index.ts:5:1)
at Module._compile (internal/modules/cjs/loader.js:678:30)
at Module.m._compile (/home/eschmidt/Workspace/cli/node_modules/ts-node/src/index.ts:403:23)
at Module._extensions..js (internal/modules/cjs/loader.js:689:10)
at Object.require.extensions.(anonymous function) [as .ts] (/home/eschmidt/Workspace/cli/node_modules/ts-node/src/index.ts:406:12)
at Module.load (internal/modules/cjs/loader.js:589:32)
module: @oclif/config@1.6.17
task: toCached
plugin: my-plugin
root: /home/eschmidt/Workspace/cli
我无法理解的是,当我 运行 tsc --traceResolution
模块正确解析时:
======== Module name 'my-module' was successfully resolved to '/home/eschmidt/Workspace/cli/src/my-module.ts'. ========
我的 tsconfig.json
文件包含:
{
"compilerOptions": {
"declaration": true,
"moduleResolution": "node",
"forceConsistentCasingInFileNames": true,
"importHelpers": true,
"module": "commonjs",
"sourceMap": true,
"outDir": "./lib",
"pretty": true,
"rootDirs": [
"./src/"
],
"strict": true,
"target": "es2017",
"baseUrl": "src"
},
"include": [
"./src/**/*"
]
}
如果有人能阐明这个问题,或者至少建议在哪里寻求进一步的帮助,我将不胜感激。如果需要更多详细信息,请告诉我。
提前致谢!
原来问题是因为虽然tsc和ts-node都使用baseUrl
进行绝对路径解析,但它们都没有执行任何类型的从绝对路径到相对路径的实际映射生成的 Javascript 代码。换句话说,转译后的 JS 文件和 ts-node 内部生成的代码最终都具有:
import {MyClass} = require('my-module')
而我希望它们包含类似的内容:
import {MyClass} = require('../my-module')
这阻止了节点的模块加载器找到模块。 ts-node 也没有工作,我相信,因为根本没有 tsconfig.json 文件来指示路径映射。
尽管 IMO 令人困惑,并且没有正确记录,但正如所讨论的那样,这是预期的行为 here. As of now, absolute to relative path mapping is not supported by Typescript (see https://github.com/Microsoft/TypeScript/issues/15479)。
为了避免称为路径地狱的情况,这意味着具有非常深的相对导入路径,我发现 module-alias and tsmodule-alias 非常有用。这些模块改变模块加载器的行为,以便它自动将别名映射到相对路径。
有关此问题的更多信息,请参阅 Github 上的 issue。
另一个可能相关的解决方案是 运行 节点与 NODE_PATH=dist node dist/index.js
。这实质上指定了 node
每个绝对导入(相对于 baseUrl
)应该使用的路径
我有一个用 Typescript 编写的 Node.js 项目,预计 运行 作为 CLI,但我无法使用 node_modules
目录导入模块绝对路径(相对路径工作正常)。值得一提的是,我正在使用 oclif framework 构建我的 CLI。
我的项目组织如下:
cli
|--node_modules
|--src
|--my-module.ts
|--subdir
|--index.ts
在 my-module.ts
我有:
export class MyClass {
myClassFcn(s: string) {
return 'result'
}
}
index.ts
脚本包含如下内容:
import {MyClass} = require('my-module')
当我尝试使用 ts-node 执行我的应用程序时,我得到
(node:10423) [MODULE_NOT_FOUND] Error Plugin: cli: Cannot find module 'my-module'
module: @oclif/config@1.6.17
task: toCached
plugin: cli
root: /home/eschmidt/Workspace/cli
Error Plugin: cli: Cannot find module 'my-module'
at Function.Module._resolveFilename (internal/modules/cjs/loader.js:571:15)
at Function.Module._load (internal/modules/cjs/loader.js:497:25)
at Module.require (internal/modules/cjs/loader.js:626:17)
at require (internal/modules/cjs/helpers.js:20:18)
at Object.<anonymous> (/home/eschmidt/Workspace/cli/src/commands/create/index.ts:5:1)
at Module._compile (internal/modules/cjs/loader.js:678:30)
at Module.m._compile (/home/eschmidt/Workspace/cli/node_modules/ts-node/src/index.ts:403:23)
at Module._extensions..js (internal/modules/cjs/loader.js:689:10)
at Object.require.extensions.(anonymous function) [as .ts] (/home/eschmidt/Workspace/cli/node_modules/ts-node/src/index.ts:406:12)
at Module.load (internal/modules/cjs/loader.js:589:32)
module: @oclif/config@1.6.17
task: toCached
plugin: my-plugin
root: /home/eschmidt/Workspace/cli
我无法理解的是,当我 运行 tsc --traceResolution
模块正确解析时:
======== Module name 'my-module' was successfully resolved to '/home/eschmidt/Workspace/cli/src/my-module.ts'. ========
我的 tsconfig.json
文件包含:
{
"compilerOptions": {
"declaration": true,
"moduleResolution": "node",
"forceConsistentCasingInFileNames": true,
"importHelpers": true,
"module": "commonjs",
"sourceMap": true,
"outDir": "./lib",
"pretty": true,
"rootDirs": [
"./src/"
],
"strict": true,
"target": "es2017",
"baseUrl": "src"
},
"include": [
"./src/**/*"
]
}
如果有人能阐明这个问题,或者至少建议在哪里寻求进一步的帮助,我将不胜感激。如果需要更多详细信息,请告诉我。
提前致谢!
原来问题是因为虽然tsc和ts-node都使用baseUrl
进行绝对路径解析,但它们都没有执行任何类型的从绝对路径到相对路径的实际映射生成的 Javascript 代码。换句话说,转译后的 JS 文件和 ts-node 内部生成的代码最终都具有:
import {MyClass} = require('my-module')
而我希望它们包含类似的内容:
import {MyClass} = require('../my-module')
这阻止了节点的模块加载器找到模块。 ts-node 也没有工作,我相信,因为根本没有 tsconfig.json 文件来指示路径映射。
尽管 IMO 令人困惑,并且没有正确记录,但正如所讨论的那样,这是预期的行为 here. As of now, absolute to relative path mapping is not supported by Typescript (see https://github.com/Microsoft/TypeScript/issues/15479)。
为了避免称为路径地狱的情况,这意味着具有非常深的相对导入路径,我发现 module-alias and tsmodule-alias 非常有用。这些模块改变模块加载器的行为,以便它自动将别名映射到相对路径。
有关此问题的更多信息,请参阅 Github 上的 issue。
另一个可能相关的解决方案是 运行 节点与 NODE_PATH=dist node dist/index.js
。这实质上指定了 node
每个绝对导入(相对于 baseUrl
)应该使用的路径