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 将始终失败并报错。
当你的包作为依赖被安装时,除非用户已设置 engine-strict
配置标志 (参阅 .npmrc),否则此字段仅供参考,且只会产生警告。
dependenciesMeta
用于在 dependencies
, optionalDependencies
和 devDependencies
中声明的依赖项的补充元信息。
dependenciesMeta.*.injected
如果将本地工作区包依赖项设置为 true
,则将通过在虚拟存储中创建硬链接副本来安装该包(node_modules/.pnpm
)。
如果将其设置为 false
或未设置,则依赖项将通过创建指向工作区中包的源目录的 node_modules
符号链接来安装。 这是缺省值,因为它更快,并确保对依赖关系的任何修改都将立即对其使用者可视。
例如,假设以下 package.json
是一个本地工作区包:
{
"name": "card",
"dependencies": {
"button": "workspace:1.0.0"
}
}
button
依赖项通常会通过在 card
的 node_modules
目录中创建符号链接来安装,该符号链接指向 button
的开发目录。
但是,如果 button
在其 peerDependencies
中指定 react
,该怎么办? 如果 monorepo 中的所有项目都使用相同版本的 react
,那么就没有问题。 但如果 card
依赖的 button
使用 react@16
并且 form
使用 react@17
呢? 通常你必须选择一个版本的 react
并使用 button
的 devDependencies
来指定它。 符号链接无法为 react
对等依赖提供一种方式,使不同的消费者(例如 card
和 form
)能够以不同的方式满足它。
injected
字段通过在虚拟存储中安装 button
的硬链接副本解决了这个问题。 为了实现这一点, card
的 package.json
可以按如下方式配置:
{
"name": "card",
"dependencies": {
"button": "workspace:1.0.0",
"react": "16"
},
"dependenciesMeta": {
"button": {
"injected": true
}
}
}
而 form
的 package.json
可以按如下方式配置:
{
"name": "form",
"dependencies": {
"button": "workspace:1.0.0",
"react": "17"
},
"dependenciesMeta": {
"button": {
"injected": true
}
}
}
通过这些变化,我们说 button
是 card
和 form
的“注入依赖项”。 当 button
导入 react
时,它将在 card
的上下文中解析为 react@16
,但在 form
的上下文中解析为 react@17
。
由于注入依赖关系会生成其工作空间源目录的副本,因此每当修改代码时,都必须更新这些副本; 否则,新状态将不会反映给使用者。 当使用命令(例如 pnpm --recursive run build
)构建多个项目时,此更新必须在重建每个注入的包之后但重建其使用者之前进行。 对于简单的用例,可以通过再次调用 pnpm install
来完成,也许是使用 package.json
生命周期脚本(如 "prepare": "pnpm run build"
来重建该项目)。 第三方工具,如 pnpm-sync 和 pnpm-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
}
}
pnpm.overrides
此字段允许你指示 pnpm 覆盖依赖关系图中的任何依赖项。 这对于强制所有软件包使用一个依赖项的单个版本、反向移植一个修复、用分叉替换依赖项或删除未使用的依赖项很有用。
请注意,overrides 字段只能在项目的根目录下设置。
"pnpm"."overrides"
字段的示例:
{
"pnpm": {
"overrides": {
"foo": "^1.0.0",
"quux": "npm:@myorg/quux@^1.0.0",
"bar@^2.1.0": "3.0.0",
"qar@1>zoo": "2"
}
}
}
你可以用 ">" 来覆盖某个包下的子依赖的版本,比如 qar@1>zoo
只会覆盖 qar@1
依赖的 zoo
的版本,而不会影响其他依赖。
一个覆盖可以被定义为直接依赖项的规则的引用。 这通过在依赖名称前缀一个 $
实现:
{
"dependencies": {
"foo": "^1.0.0"
},
"pnpm": {
"overrides": {
"foo": "$foo"
}
}
}
被引用的包不必匹配需要覆盖的包:
{
"dependencies": {
"foo": "^1.0.0"
},
"pnpm": {
"overrides": {
"bar": "$foo"
}
}
}
如果你发现使用某个包不需要它的某个依赖项,则可以使用 -
将其删除。 例如,如果软件包 foo@1.0.0
需要一个名为 bar
的大型软件包来实现你不使用的功能,则删除它可以减少安装时间:
{
"pnpm": {
"overrides": {
"foo@1.0.0>bar": "-"
}
}
}
此功能对于 optionalDependencies
特别有用,其中大多数可选包可以被安全地跳过。
pnpm.packageExtensions
packageExtension
字段提供了一种用额外信息扩展现有软件包定义的方法。 例如,如果 react-redux
本应该在它的 peerDependencies
中包含 react-dom
但它没有,则可以用 packageExtensions
来填补上 react-redux
。
{
"pnpm": {
"packageExtensions": {
"react-redux": {
"peerDependencies": {
"react-dom": "*"
}
}
}
}
}
packageExtensions
中的键是包名或报名和 semver 范围,因此可以只修改包的某些版本:
{
"pnpm": {
"packageExtensions": {
"react-redux@1": {
"peerDependencies": {
"react-dom": "*"
}
}
}
}
}
以下字段可以使用 packageExtensions
被扩展:dependencies
optionalDependencies
peerDependencies
和 peerDependenciesMeta
。
一个更大的例子:
{
"pnpm": {
"packageExtensions": {
"express@1": {
"optionalDependencies": {
"typescript": "2"
}
},
"fork-ts-checker-webpack-plugin": {
"dependencies": {
"@babel/core": "1"
},
"peerDependencies": {
"eslint": ">= 6"
},
"peerDependenciesMeta": {
"eslint": {
"optional": true
}
}
}
}
}
}
我们与 Yarn 一起维护一个 packageExtensions
的数据库,以便修补在生态系统中损坏的包。 如果你使用 packageExtensions
, 考虑发送一个 PR 上游并将你的扩展贡献给 @yarnpkg/extension
数据库。
pnpm.peerDependencyRules
pnpm.peerDependencyRules.ignoreMissing
pnpm 不会打印有关依赖列表中缺少对 peerDependency 的警告。
例如,使用以下配置,如果依赖项需要 react
但 react
未被安装 ,pnpm 不会打印相应警告。
{
"pnpm": {
"peerDependencyRules": {
"ignoreMissing": ["react"]
}
}
}
包名也可以使用模式匹配
{
"pnpm": {
"peerDependencyRules": {
"ignoreMissing": ["@babel/*", "@eslint/*"]
}
}
}