跳到主内容
版本:Next

package.json

一个包的清单文件。 它包含包的所有元数据,包括依赖项、标题、作者等等。 这是所有主要的 Node.js 包管理工具,包括 pnpm 的保留标准。

除了传统的 package.json 格式之外,pnpm 还支持 package.json5 (通过 json5)和 package.yaml (通过 js-yaml)。

engines

你可以指定你的软件能够运行的 Node 版本和 pnpm 版本:

{
"engines": {
"node": ">=10",
"pnpm": ">=3"
}
}

在本地开发时, 如果其版本与 engines 字段中指定的版本不匹配,pnpm 将始终失败并报错。

除非用户设置了 engineStrict 配置标志(参见 设置),否则此 字段仅供参考,并且仅当您的包作为依赖项安装时才会产生警告。

devEngines.runtime

添加于:v10.14

允许指定项目使用的一个或多个 JavaScript 运行时引擎。 支持的运行时包括 Node.js、Deno 和 Bun。

例如,下面是如何将 node@^24.4.0 添加到您的依赖项中:

{
"devEngines": {
"runtime": {
"name": "node",
"version": "^24.4.0",
"onFail": "download"
}
}
}

您还可以将多个运行时添加到同一个 package.json

{
"devEngines": {
"runtime": [
{
"name": "node",
"version": "^24.4.0",
"onFail": "download"
},
{
"name": "deno",
"version": "^2.4.3",
"onFail": "download"
}
]
}
}

工作原理:

  1. pnpm install 将您指定的范围解析为最新的匹配的运行时版本。
  2. 精确的版本(和校验和)保存在锁文件(lockfile)中。
  3. 脚本使用本地运行时,确保跨环境的一致性。

dependenciesMeta

用于在 dependencies, optionalDependenciesdevDependencies 中声明的依赖项的补充元信息。

dependenciesMeta.*.injected

如果将本地工作区包依赖项设置为 true ,则将通过在虚拟存储中创建硬链接副本来安装该包(node_modules/.pnpm)。

如果将其设置为 false 或未设置,则依赖项将通过创建指向工作区中包的源目录的 node_modules 符号链接来安装。 这是缺省值,因为它更快,并确保对依赖关系的任何修改都将立即对其使用者可视。

例如,假设以下 package.json 是一个本地工作区包:

{
"name": "card",
"dependencies": {
"button": "workspace:1.0.0"
}
}

button 依赖项通常会通过在 cardnode_modules 目录中创建符号链接来安装,该符号链接指向 button 的开发目录。

但是,如果 button 在其 peerDependencies 中指定 react,该怎么办? 如果 monorepo 中的所有项目都使用相同版本的 react,那么就没有问题。 但如果 card 依赖的 button 使用 react@16 并且 form 使用 react@17 呢? 通常你必须选择一个版本的 react 并使用 buttondevDependencies 来指定它。 符号链接无法为 react 对等依赖提供一种方式,使不同的消费者(例如 cardform)能够以不同的方式满足它。

injected 字段通过在虚拟存储中安装 button 的硬链接副本解决了这个问题。 为了实现这一点, cardpackage.json 可以按如下方式配置:

{
"name": "card",
"dependencies": {
"button": "workspace:1.0.0",
"react": "16"
},
"dependenciesMeta": {
"button": {
"injected": true
}
}
}

formpackage.json 可以按如下方式配置:

{
"name": "form",
"dependencies": {
"button": "workspace:1.0.0",
"react": "17"
},
"dependenciesMeta": {
"button": {
"injected": true
}
}
}

通过这些变化,我们说 buttoncardform 的“注入依赖项”。 当 button 导入 react 时,它将在 card 的上下文中解析为 react@16,但在 form 的上下文中解析为 react@17

由于注入依赖关系会生成其工作空间源目录的副本,因此每当修改代码时,都必须更新这些副本; 否则,新状态将不会反映给使用者。 当使用命令(例如 pnpm --recursive run build)构建多个项目时,此更新必须在重建每个注入的包之后但重建其使用者之前进行。 对于简单的用例,可以通过再次调用 pnpm install 来完成,也许是使用 package.json 生命周期脚本(如 "prepare": "pnpm run build" 来重建该项目)。 第三方工具,如 pnpm-syncpnpm-sync-dependencies-meta-injected 为更新注入的依赖项以及监视模式支持提供了更为强大和高效的解决方案。

peerDependenciesMeta

此字段列出了一些与 peerDependencies 字段中列出的依赖关系相关的额外信息。

peerDependenciesMeta.*.optional

如果设置为 true,所选的 peer dependency 将被包管理工具标记为可选的。 因此,消费方省略它将不再被报告为错误。

示例:

{
"peerDependencies": {
"foo": "1"
},
"peerDependenciesMeta": {
"foo": {
"optional": true
},
"bar": {
"optional": true
}
}
}

请注意,即使在 peerDependencies 中没有指定 bar,它也会被标记为可选的。 因此,pnpm 将假定任何版本的 bar 都是被允许的。 但是,foo 是可选的,但只能使用指定的版本。

publishConfig

在包被打包之前,可以覆盖清单中的某些字段。 以下字段可以被覆盖:

要覆盖字段,请将字段的要发布的版本添加到 publishConfig

例如,以下 package.json

{
"name": "foo",
"version": "1.0.0",
"main": "src/index.ts",
"publishConfig": {
"main": "lib/index.js",
"typings": "lib/index.d.ts"
}
}

将被发布为:

{
"name": "foo",
"version": "1.0.0",
"main": "lib/index.js",
"typings": "lib/index.d.ts"
}

publishConfig.executableFiles

默认情况下,出于可移植性的原因,除了 bin 字段中列出的文件之外,不会在生成的包存档中将任何文件标记为可执行文件。 executableFiles 字段允许你声明必须设置可执行标志 (+x) 的额外字段,即使它们不能通过 bin 字段直接访问。

{
"publishConfig": {
"executableFiles": [
"./dist/shim.js"
]
}
}

publishConfig.directory

你还可以使用 publishConfig.directory 字段来自定义相对于当前 package.json 的发布子目录。

预计在指定目录中有当前包的修改版本(通常使用第三方构建工具)。

在这个例子中 "dist" 文件夹必须包含一个 package.json

{
"name": "foo",
"version": "1.0.0",
"publishConfig": {
"directory": "dist"
}
}

publishConfig.linkDirectory

  • 默认值: true
  • 类型:Boolean

当设置为 true 时,项目将在本地开发期间从 publishConfig.directory 位置进行符号链接。

示例:

{
"name": "foo",
"version": "1.0.0",
"publishConfig": {
"directory": "dist",
"linkDirectory": true
}
}