Splitting up large codebases into separate independently versioned packages
is extremely useful for code sharing. However, making changes across many
repositories is messy and difficult to track, and testing across repositories
gets complicated really fast.
把一个大型代码库拆分成相互分离独立的版本包,对于代码共享来说是非常有帮助的。
To solve these (and many other) problems, some projects will organize their
codebases into multi-package repositories (sometimes called monorepos). Projects like Babel, React, Angular,
Ember, Meteor, Jest, and many others develop all of their packages within a single repository.
Lerna is a tool that optimizes the workflow around managing multi-package
repositories with git and npm.
Lerna 是一个用git和npm来优化这个管理多包库工作流的工具
Lerna can also reduce the time and space requirements for numerous
copies of packages in development and build environments - normally a
downside of dividing a project into many separate NPM package. See the
hoist documentation for details.
The two primary commands in Lerna are lerna bootstrap and lerna publish.
Lerna 的两个主要命令是 lerna boostrap 和 lerna publish。
bootstrap will link dependencies in the repo together.
publish will help publish any updated packages.
bootstrap 会把库的依赖链接在一起
publish 会把任何一个有变更的package发布
Getting Started
开始
The instructions below are for Lerna 2.x.
We recommend using it instead of 1.x for a new Lerna project. Check the wiki if you need to see the 1.x README.
Let's start by installing Lerna globally with npm.
使用 npm 全局安装Lerna,并开始使用。
$ npm install --global lerna
Next we'll create a new folder:
创建一个新的文件夹
$ mkdir lerna-repo
$ cd lerna-repo
And now let's turn it into a Lerna repo:
把文件夹变成 Lerna 库
$ lerna init
This will create a lerna.json configuration file as well as a packages folder, so your folder should now look like this:
创建了一个和 package文件类似的 lerna.json配置配置文件,现在的文件夹应该长这样:
lerna-repo/
packages/
package.json
lerna.json
How It Works
Lerna 如何工作
Lerna allows you to manage your project using one of two modes: Fixed or Independent.
Lerna 有两种模式去管理项目: 固定模式,相互独立模式
Fixed/Locked mode (default)
固定/锁定模式(默认)
Fixed mode Lerna projects operate on a single version line. The version is kept in the lerna.json file at the root of your project under the version key. When you run lerna publish, if a module has been updated since the last time a release was made, it will be updated to the new version you're releasing. This means that you only publish a new version of a package when you need to.
固定模式的Lerna项目是在一条单一的版本线上操作的,这个版本是在项目根目录的 lerna.json文件中的 version字段中控制的。当执行 lerna publish时,如果一个模块自上次发布以来有更新,它将会更新为你要发布的新的版本,这意味着你只需在需要时发布新版本的package。
This is the mode that Babel is currently using. Use this if you want to automatically tie all package versions together. One issue with this approach is that a major change in any package will result in all packages having a new major version.
这是目前Babel在用的模式。如果你想自动把所有依赖包版本捆绑在一起,请使用这种模式。这种方法的一个问题是,任何package的一个重大改动,都会导致所有的package会有一个新的主要版本。
Independent mode (--independent)
独立模式(--independent)
Independent mode Lerna projects allows maintainers to increment package versions independently of each other. Each time you publish, you will get a prompt for each package that has changed to specify if it's a patch, minor, major or custom change.
独立模式的Lerna允许维护者相互独立地变更package的版本。你会得到每个已变更的package的提示,以指定它是补丁,次要的,重要的自定义的修改。
Independent mode allows you to more specifically update versions for each package and makes sense for a group of components. Combining this mode with something like semantic-release would make it less painful. (There is work on this already at atlassian/lerna-semantic-release).
独立模式允许你更具体地更新各个package的版本,并对一组组件更加有意义。将这种模式和一些像 semantic-release相结合起来会没有那么困难。
The version key in lerna.json is ignored in independent mode.
在独立模式中 lerna.json文件中的version字段会被忽略
Troubleshooting
故障排除
If you encounter any issues while using Lerna please check out our Troubleshooting document where you might find the answer to your problem.
如果在使用Lerna过程中,没有找到任何有帮助的issues,请查看Troubleshooting文档,这里应该可以找到问题的答案。
Create a new Lerna repo or upgrade an existing repo to the current version of Lerna.
创建一个Lerna库或用现在的Lerna版本升级一个已存在的库
Lerna assumes the repo has already been initialized with git init.
Lerna 默认假设你的库已经关联git,既已经有默认的git init命令
When run, this command will:
Add lerna as a devDependency in package.json if it doesn't already exist.
Create a lerna.json config file to store the version number.
运行时,这个命令会:
如果package.json中不存在lerna,就添加进去;
创建一个lerna.json配置文件来存储version号码;
Example output on a new git repo:
$ lerna init
lerna info version v2.0.0
lerna info Updating package.json
lerna info Creating lerna.json
lerna success Initialized Lerna files
--independent, -i
$ lerna init --independent
This flag tells Lerna to use independent versioning mode.
这个标识是设置Lerna用独立版本模式
--exact
$ lerna init --exact
By default, lerna init will use a caret range when adding or updating the local version of lerna, just like npm install --save-dev lerna.
默认情况下,lerna init 会在添加或更新本地lerna版本时使用插入符号范围的命令,就像 npm install --save-dev lerna
To retain the lerna 1.x behavior of "exact" comparison, pass this flag.
It will configure lerna.json to enforce exact match for all subsequent executions.
要保留lerna 1.x的"exact"行为,请传递这个标识,它会配置 lerna.json去执行后续全部的完全执行。
Bootstrap the packages in the current Lerna repo.
Installs all of their dependencies and links any cross-dependencies.
引导目前Lerna库的所有package,安装它们全部的依赖关系并连接任何相互交叉依赖的关系。
When run, this command will:
当这个命令执行时,会发生下面的情况:
npm install all external dependencies of each package.
Symlink together all Lerna packages that are dependencies of each other.
npm run prepublish in all bootstrapped packages.
npm run prepare in all bootstrapped packages.
npm install 安装每个package的额外依赖;
把所有相互有依赖的Lerna packages 链接起来;
npm run prepublish 在所有有联系的包中执行;
npm run prepare 在所有有联系的包中执行;
lerna bootstrap respects the --ignore, --ignore-scripts, --scope and --include-filtered-dependencies flags (see Flags).
Pass extra arguments to npm client by placing them after --:
通过将它们放在‘--’后面,将额外的参数传递给npm客户端;
lerna add respects the --ignore, --scope and --include-filtered-dependencies flags (see Flags).
lerna add 可以有--ignore, --scope and --include-filtered-dependencies这些标识
Examples
lerna add module-1 --scope=module-2 # Install module-1 to module-2
lerna add module-1 --scope=module-2 --dev # Install module-1 to module-2 in devDependencies
lerna add module-1 # Install module-1 in all modules except module-1
lerna add babel-core # Install babel-core in all modules
Lerna checks if each dependency is also part of the Lerna repo.
In this example, babel-generator can be an internal dependency, while source-map is always an external dependency.
The version of babel-generator in the package.json of babel-core is satisfied by packages/babel-generator, passing for an internal dependency.
source-map is npm installed (or yarned) like normal.
packages/babel-core/node_modules/babel-generator symlinks to packages/babel-generator
This allows nested directory imports
Notes:
When a dependency version in a package is not satisfied by a package of the same name in the repo, it will be npm installed (or yarned) like normal.
Dist-tags, like latest, do not satisfy semver ranges.
Circular dependencies result in circular symlinks which may impact your editor/IDE.
Webstorm locks up when circular symlinks are present. To prevent this, add node_modules to the list of ignored files and folders in Preferences | Editor | File Types | Ignored files and folders.
publish
$ lerna publish
Publish packages in the current Lerna project. When run, this command does the following:
Creates a new release of the packages that have been updated.
Prompts for a new version.
Creates a new git commit/tag in the process of publishing to npm.
More specifically, this command will:
Run the equivalent of lerna updated to determine which packages need to be published.
If necessary, increment the version key in lerna.json.
Update the package.json of all updated packages to their new versions.
Update all dependencies of the updated packages with the new versions, specified with a caret (^).
Create a new git commit and tag for the new version.
Publish updated packages to npm.
Lerna won't publish packages which are marked as private ("private": true in the package.json).
Note: to publish scoped packages, you need to add the following to each package.json:
"publishConfig": {
"access": "public"
}
--exact
$ lerna publish --exact
When run with this flag, publish will specify updated dependencies in updated packages exactly (with no punctuation), instead of as semver compatible (with a ^).
For more information, see the package.json dependencies documentation.
--npm-tag [tagname]
$ lerna publish --npm-tag=next
When run with this flag, publish will publish to npm with the given npm dist-tag (defaults to latest).
This option can be used to publish a prerelease or beta version.
Note: the latest tag is the one that is used when a user runs npm install my-package.
To install a different tag, a user can run npm install my-package@prerelease.
When run with this flag, publish publishes packages in a more granular way (per commit). Before publishing to npm, it creates the new version tag by taking the current version, bumping it to the next minor version, adding the provided meta suffix (defaults to alpha) and appending the current git sha (ex: 1.0.0 becomes 1.1.0-alpha.81e3b443).
The intended use case for this flag is a per commit level release or nightly release.
By default, the changelog preset is set to angular.
In some cases you might want to change either use a another preset or a custom one.
Presets are names of built-in or installable configuration for conventional changelog.
Presets may be passed as the full name of the package, or the auto-expanded suffix
(e.g., angular is expanded to conventional-changelog-angular).
--git-remote [remote]
$ lerna publish --git-remote upstream
When run with this flag, publish will push the git changes to the specified remote instead of origin.
--skip-git
$ lerna publish --skip-git
When run with this flag, publish will publish to npm without running any of the git commands.
Only publish to npm; skip committing, tagging, and pushing git changes (this only affects publish).
--skip-npm
$ lerna publish --skip-npm
When run with this flag, publish will update all package.json package
versions and dependency versions, but it will not actually publish the
packages to npm.
This flag can be combined with --skip-git to just update versions and
dependencies, without committing, tagging, pushing or publishing.
Only update versions and dependencies; don't actually publish (this only affects publish).
--force-publish [packages]
$ lerna publish --force-publish=package-2,package-4
# force publish all packages
$ lerna publish --force-publish=*
When run with this flag, publish will force publish the specified packages (comma-separated) or all packages using *.
This will skip the lerna updated check for changed packages and forces a package that didn't have a git diff change to be updated.
--yes
$ lerna publish --canary --yes
# skips `Are you sure you want to publish the above changes?`
When run with this flag, publish will skip all confirmation prompts.
Useful in Continuous integration (CI) to automatically answer the publish confirmation prompt.
--cd-version
$ lerna publish --cd-version (major | minor | patch | premajor | preminor | prepatch | prerelease)
# uses the next semantic version(s) value and this skips `Select a new version for...` prompt
When run with this flag, publish will skip the version selection prompt (in independent mode) and use the next specified semantic version.
You must still use the --yes flag to avoid all prompts. This is useful when build systems need to publish without command prompts. Works in both normal and independent modes.
If you have any packages with a prerelease version number (e.g. 2.0.0-beta.3) and you run lerna publish with --cd-version and a non-prerelease version increment (major / minor / patch), it will publish those packages in addition to the packages that have changed since the last release.
--preid
$ lerna publish --cd-version=prerelease
# uses the next semantic prerelease version, e.g.
# 1.0.0 => 1.0.0-0
$ lerna publish --cd-version=prepatch --preid=next
# uses the next semantic prerelease version with a specific prerelease identifier, e.g.
# 1.0.0 => 1.0.1-next.0
When run with this flag, lerna publish --cd-version will
increment premajor, preminor, prepatch, or prerelease
versions using the specified prerelease identifier.
--repo-version
$ lerna publish --repo-version 1.0.1
# applies version and skips `Select a new version for...` prompt
When run with this flag, publish will skip the version selection prompt and use the specified version.
Useful for bypassing the user input prompt if you already know which version to publish.
When run with this flag, publish will use the provided message when committing the version updates
for publication. Useful for integrating lerna into projects that expect commit messages to adhere
to certain guidelines, such as projects which use commitizen and/or semantic-release.
If the message contains %s, it will be replaced with the new global version version number prefixed with a "v".
If the message contains %v, it will be replaced with the new global version version number without the leading "v".
Note that this only applies when using the default "fixed" versioning mode, as there is no "global" version when using --independent.
Lerna allows you to specify a glob or an array of globs in your lerna.json that your current branch needs to match to be publishable.
You can use this flag to override this setting.
If your lerna.json contains something like this:
and you are not on the branch master lerna will prevent you from publishing. To force a publish despite this config, pass the --allow-branch flag:
$ lerna publish --allow-branch my-new-feature
updated
$ lerna updated
Check which packages have changed since the last release (the last git tag).
Lerna determines the last git tag created and runs git diff --name-only v6.8.1 to get all files changed since that tag. It then returns an array of packages that have an updated file.
Note that configuration for the publish command also affects the
updated command. For example config.publish.ignore
--json
$ lerna updated --json
When run with this flag, updated will return an array of objects in the following format:
$ lerna run <script> -- [..args] # runs npm run my-script in all packages that have it
$ lerna run test
$ lerna run build
# watch all packages and transpile on change, streaming prefixed output
$ lerna run --parallel watch
Run an npm script in each package that contains that script. A double-dash (--) is necessary to pass dashed arguments to the script execution.
lerna run respects the --concurrency, --scope, --ignore, --stream, and --parallel flags (see Flags).
$ lerna run --scope my-component test
Note: It is advised to constrain the scope of this command (and lerna exec,
below) when using the --parallel flag, as spawning dozens of subprocesses
may be harmful to your shell's equanimity (or maximum file descriptor limit,
for example). YMMV
exec
$ lerna exec -- <command> [..args] # runs the command in all packages
$ lerna exec -- rm -rf ./node_modules
$ lerna exec -- protractor conf.js
Run an arbitrary command in each package.
A double-dash (--) is necessary to pass dashed flags to the spawned command, but is not necessary when all the arguments are positional.
lerna exec respects the --concurrency, --scope, --ignore, --stream and --parallel flags (see Flags).
$ lerna exec --scope my-component -- ls -la
To spawn long-running processes, pass the --parallel flag:
# transpile all modules as they change in every package
$ lerna exec --parallel -- babel src -d lib -w
You may also get the name of the current package through the environment variable LERNA_PACKAGE_NAME:
$ lerna exec -- npm view \$LERNA_PACKAGE_NAME
You may also run a script located in the root dir, in a complicated dir structure through the environment variable LERNA_ROOT_PATH:
Hint: The commands are spawned in parallel, using the concurrency given (except with --parallel).
The output is piped through, so not deterministic.
If you want to run the command in one package after another, use it like this:
$ lerna exec --concurrency 1 -- ls -la
--bail
$ lerna exec --bail=<boolean> <command>
This flag signifies whether or not the exec command should halt execution upon encountering an error thrown by one of the spawned subprocesses. Its default value is true.
import
$ lerna import <path-to-external-repository>
Import the package at <path-to-external-repository>, with commit history,
into packages/<directory-name>. Original commit authors, dates and messages
are preserved. Commits are applied to the current branch.
This is useful for gathering pre-existing standalone packages into a Lerna
repo. Each commit is modified to make changes relative to the package
directory. So, for example, the commit that added package.json will
instead add packages/<directory-name>/package.json.
link
$ lerna link
Symlink together all Lerna packages that are dependencies of each other in the current Lerna repo.
--force-local
$ lerna link --force-local
When passed, this flag causes the link command to always symlink local dependencies regardless of matching version range.
Misc
Lerna will log to a lerna-debug.log file (same as npm-debug.log) when it encounters an error running a command.
commands.publish.ignore: an array of globs that won't be included in lerna updated/publish. Use this to prevent publishing a new version unnecessarily for changes, such as fixing a README.md typo.
commands.bootstrap.ignore: an array of globs that won't be bootstrapped when running the lerna bootstrap command.
commands.bootstrap.scope: an array of globs that restricts which packages will be bootstrapped when running the lerna bootstrap command.
packages: Array of globs to use as package locations.
Common devDependencies
Most devDependencies can be pulled up to the root of a Lerna repo.
This has a few benefits:
All packages use the same version of a given dependency
Can keep dependencies at the root up-to-date with an automated tool such as GreenKeeper
Dependency installation time is reduced
Less storage is needed
Note that devDependencies providing "binary" executables that are used by
npm scripts still need to be installed directly in each package where they're
used.
For example the nsp dependency is necessary in this case for lerna run nsp
(and npm run nsp within the package's directory) to work correctly:
Lerna allows target versions of local dependent packages to be written as a git remote url with a committish (e.g., #v1.0.0 or #semver:^1.0.0) instead of the normal numeric version range.
This allows packages to be distributed via git repositories when packages must be private and a private npm registry is not desired.
Please note that lerna does not perform the actual splitting of git history into the separate read-only repositories. This is the responsibility of the user. (See this comment for implementation details)
lerna bootstrap will properly symlink pkg-2 into pkg-1.
lerna publish will update the committish (#v1.0.0) in pkg-1 when pkg-2 changes.
Flags
Options to Lerna can come from configuration (lerna.json) or on the command
line. Additionally options in config can live at the top level or may be
applied to specific commands.
In this case exampleOption will be "foo" for all commands except init,
where it will be "bar". In all cases it may be overridden to "baz" on the
command-line with --example-option=baz.
--concurrency
How many threads to use when Lerna parallelizes the tasks (defaults to 4)
$ lerna publish --concurrency 1
--scope [glob]
Scopes a command to a subset of packages.
$ lerna exec --scope my-component -- ls -la
$ lerna run --scope toolbar-* test
--since [ref]
When executing a script or command, scope the operation to packages that have been updated since the specified ref. If ref is not specified, it defaults to the latest tag.
List the contents of packages that have changed since the latest tag:
$ lerna exec --since -- ls -la
Run the tests for all packages that have changed since master:
$ lerna run test --since master
List all packages that have changed since some-branch:
$ lerna ls --since some-branch
This can be particularly useful when used in CI, if you can obtain the target branch a PR will be going into, because you can use that as the ref to the --since option. This works well for PRs going into master as well as feature branches.
--flatten
When importing repositories with merge commits with conflicts, the import command will fail trying to apply all commits. The user can use this flag to ask for import of "flat" history, i.e. with each merge commit as a single change the merge introduced.
$ lerna import ~/Product --flatten
--ignore [glob]
Excludes a subset of packages when running a command.
$ lerna bootstrap --ignore component-*
The ignore flag, when used with the bootstrap command, can also be set in lerna.json under the commands.bootstrap key. The command-line flag will take precedence over this option.
Hint: The glob is matched against the package name defined in package.json,
not the directory name the package lives in.
--ignore-scripts
When used with the bootstrap command it won't run any lifecycle scripts in bootstrapped packages.
$ lerna bootstrap --ignore-scripts
--include-filtered-dependencies
Used in combination with any command that accepts --scope (bootstrap, clean, ls, run, exec). Ensures that all dependencies (and dev dependencies) of any scoped packages (either through --scope or --ignore) are operated on as well.
Note: This will override the --scope and --ignore flags.
i.e. A package matched by the --ignore flag will still be bootstrapped if it is depended on by another package that is being bootstrapped.
This is useful for situations where you want to "set up" a single package that relies on other packages being set up.
$ lerna bootstrap --scope my-component --include-filtered-dependencies
# my-component and all of its dependencies will be bootstrapped
$ lerna bootstrap --scope "package-*" --ignore "package-util-*" --include-filtered-dependencies
# all package-util's will be ignored unless they are depended upon by a
# package matched by "package-*"
What level of logs to report. On failure, all logs are written to lerna-debug.log in the current working directory.
Any logs of a higher level than the setting are shown. The default is "info".
--max-buffer [in-bytes]
Set a max buffer length for each underlying process call. Useful for example
when someone wants to import a repo with a larger amount of commits while
running lerna import. In that case the built-in buffer length might not
be sufficient.
--no-sort
By default, all tasks execute on packages in topologically sorted order as to respect the dependency relationships of the packages in question. Cycles are broken on a best-effort basis in a way not guaranteed to be consistent across Lerna invocations.
Topological sorting can cause concurrency bottlenecks if there are a small number of packages with many dependents or if some packages take a disproportionately long time to execute. The --no-sort option disables sorting, instead executing tasks in an arbitrary order with maximum concurrency.
This option can also help if you run multiple "watch" commands. Since lerna run will execute commands in topologically sorted order, it can end up waiting for a command before moving on. This will block execution when you run "watch" commands, since they typically never end. An example of a "watch" command is running babel with the --watch CLI flag.
--hoist [glob]
Install external dependencies matching glob at the repo root so they're
available to all packages. Any binaries from these dependencies will be
linked into dependent package node_modules/.bin/ directories so they're
available for npm scripts. If the option is present but no glob is given
the default is ** (hoist everything). This option only affects the
bootstrap command.
Note: If packages depend on different versions of an external dependency,
the most commonly used version will be hoisted, and a warning will be emitted.
--nohoist [glob]
Do not install external dependencies matching glob at the repo root. This
can be used to opt out of hoisting for certain dependencies.
$ lerna bootstrap --hoist --nohoist=babel-*
--npm-client [client]
Install external dependencies using [client] install. Must be an executable
that knows how to install npm dependencies.
$ lerna bootstrap --npm-client=yarn
May also be configured in lerna.json:
{
...
"npmClient": "yarn"
}
--reject-cycles
Fail immediately if a cycle is found (in bootstrap, exec, publish or run).
$ lerna bootstrap --reject-cycles
--use-workspaces
Enables integration with Yarn Workspaces (available since yarn@0.27+).
The values in the array are the commands in which Lerna will delegate operation to Yarn (currently only bootstrapping).
If --use-workspaces is true then packages will be overridden by the value from package.json/workspaces.
May also be configured in lerna.json:
This list is broadly similar to lerna's packages config (a list of globs matching directories with a package.json),
except it does not support recursive globs ("**", a.k.a. "globstars").
--stream
Stream output from child processes immediately, prefixed with the originating
package name. This allows output from different packages to be interleaved.
$ lerna run watch --stream
--parallel
Similar to --stream, but completely disregards concurrency and topological sorting, running a given command or script immediately in all matching packages with prefixed streaming output. This is the preferred flag for long-running processes such as babel src -d lib -w run over many packages.
$ lerna exec --parallel -- babel src -d lib -w
--registry [registry]
When run with this flag, forwarded npm commands will use the specified registry for your package(s).
This is useful if you do not want to explicitly set up your registry
configuration in all of your package.json files individually when e.g. using
private registries.
--temp-tag
When passed, this flag will alter the default publish process by first publishing
all changed packages to a temporary dist-tag (lerna-temp) and then moving the
new version(s) to the default dist-tag (latest).
This is not generally necessary, as Lerna will publish packages in topological
order (all dependencies before dependents) by default.
If you prefer some guidance for cli (in case you're about to start using lerna or introducing it to a new team), you might like lerna-wizard. It will lead you through a series of well-defined steps:
A tool for managing JavaScript projects with multiple packages.
About
Splitting up large codebases into separate independently versioned packages is extremely useful for code sharing. However, making changes across many repositories is messy and difficult to track, and testing across repositories gets complicated really fast.
把一个大型代码库拆分成相互分离独立的版本包,对于代码共享来说是非常有帮助的。
To solve these (and many other) problems, some projects will organize their codebases into multi-package repositories (sometimes called monorepos). Projects like Babel, React, Angular, Ember, Meteor, Jest, and many others develop all of their packages within a single repository.
为了解决这些问题(或其他问题), 一些项目将他们的代码库用多包库(也称作 monorepos)的方式组织。像 Babel,React,Angular,Meteor,Jest等其他项目都是在一个单一的代码库中开发包的。
Lerna is a tool that optimizes the workflow around managing multi-package repositories with git and npm.
Lerna 是一个用git和npm来优化这个管理多包库工作流的工具
Lerna can also reduce the time and space requirements for numerous copies of packages in development and build environments - normally a downside of dividing a project into many separate NPM package. See the hoist documentation for details.
Lerna 可以在开发和生产环境为多份复制的package减少构建时间和空间。通常一个下载将项目分离成多个npm package。
What does a Lerna repo look like?
一个 Lerna 库看起来像什么呢?
There's actually very little to it. You have a file system that looks like this: 事实上,它几乎没有什么东西,只有像下面这样的几个系统文件:
What can Lerna do?
Lerna 可以做什么?
The two primary commands in Lerna are
lerna bootstrap
andlerna publish
.Lerna 的两个主要命令是
lerna boostrap
和lerna publish
。bootstrap
will link dependencies in the repo together.publish
will help publish any updated packages.bootstrap
会把库的依赖链接在一起publish
会把任何一个有变更的package发布Getting Started
开始
Let's start by installing Lerna globally with npm. 使用 npm 全局安装Lerna,并开始使用。
Next we'll create a new folder: 创建一个新的文件夹
And now let's turn it into a Lerna repo: 把文件夹变成 Lerna 库
This will create a
lerna.json
configuration file as well as apackages
folder, so your folder should now look like this: 创建了一个和package
文件类似的lerna.json
配置配置文件,现在的文件夹应该长这样:How It Works
Lerna 如何工作
Lerna allows you to manage your project using one of two modes: Fixed or Independent. Lerna 有两种模式去管理项目: 固定模式,相互独立模式
Fixed/Locked mode (default)
固定/锁定模式(默认)
Fixed mode Lerna projects operate on a single version line. The version is kept in the
lerna.json
file at the root of your project under theversion
key. When you runlerna publish
, if a module has been updated since the last time a release was made, it will be updated to the new version you're releasing. This means that you only publish a new version of a package when you need to. 固定模式的Lerna项目是在一条单一的版本线上操作的,这个版本是在项目根目录的lerna.json
文件中的version
字段中控制的。当执行lerna publish
时,如果一个模块自上次发布以来有更新,它将会更新为你要发布的新的版本,这意味着你只需在需要时发布新版本的package。This is the mode that Babel is currently using. Use this if you want to automatically tie all package versions together. One issue with this approach is that a major change in any package will result in all packages having a new major version. 这是目前Babel在用的模式。如果你想自动把所有依赖包版本捆绑在一起,请使用这种模式。这种方法的一个问题是,任何package的一个重大改动,都会导致所有的package会有一个新的主要版本。
Independent mode (
--independent
)独立模式(
--independent
)Independent mode Lerna projects allows maintainers to increment package versions independently of each other. Each time you publish, you will get a prompt for each package that has changed to specify if it's a patch, minor, major or custom change. 独立模式的Lerna允许维护者相互独立地变更package的版本。你会得到每个已变更的package的提示,以指定它是补丁,次要的,重要的自定义的修改。
Independent mode allows you to more specifically update versions for each package and makes sense for a group of components. Combining this mode with something like semantic-release would make it less painful. (There is work on this already at atlassian/lerna-semantic-release). 独立模式允许你更具体地更新各个package的版本,并对一组组件更加有意义。将这种模式和一些像 semantic-release相结合起来会没有那么困难。
Troubleshooting
故障排除
If you encounter any issues while using Lerna please check out our Troubleshooting document where you might find the answer to your problem. 如果在使用Lerna过程中,没有找到任何有帮助的issues,请查看Troubleshooting文档,这里应该可以找到问题的答案。
Frequently asked questions
常见问题
See FAQ.md.
Commands
命令
init
Create a new Lerna repo or upgrade an existing repo to the current version of Lerna. 创建一个Lerna库或用现在的Lerna版本升级一个已存在的库
When run, this command will:
lerna
as adevDependency
inpackage.json
if it doesn't already exist.lerna.json
config file to store theversion
number.运行时,这个命令会:
package.json
中不存在lerna
,就添加进去;lerna.json
配置文件来存储version
号码;Example output on a new git repo:
--independent, -i
This flag tells Lerna to use independent versioning mode. 这个标识是设置Lerna用独立版本模式
--exact
By default,
lerna init
will use a caret range when adding or updating the local version oflerna
, just likenpm install --save-dev lerna
. 默认情况下,lerna init
会在添加或更新本地lerna
版本时使用插入符号范围的命令,就像npm install --save-dev lerna
To retain the
lerna
1.x behavior of "exact" comparison, pass this flag. It will configurelerna.json
to enforce exact match for all subsequent executions. 要保留lerna
1.x的"exact"行为,请传递这个标识,它会配置lerna.json
去执行后续全部的完全执行。bootstrap
Bootstrap the packages in the current Lerna repo. Installs all of their dependencies and links any cross-dependencies. 引导目前Lerna库的所有package,安装它们全部的依赖关系并连接任何相互交叉依赖的关系。
When run, this command will: 当这个命令执行时,会发生下面的情况:
npm install
all external dependencies of each package.Symlink together all Lerna
packages
that are dependencies of each other.npm run prepublish
in all bootstrapped packages.npm run prepare
in all bootstrapped packages.npm install
安装每个package的额外依赖;把所有相互有依赖的Lerna packages 链接起来;
npm run prepublish
在所有有联系的包中执行;npm run prepare
在所有有联系的包中执行;lerna bootstrap
respects the--ignore
,--ignore-scripts
,--scope
and--include-filtered-dependencies
flags (see Flags).Pass extra arguments to npm client by placing them after
--
: 通过将它们放在‘--’后面,将额外的参数传递给npm客户端;May also be configured in
lerna.json
:How
bootstrap
worksLet's use
babel
as an example.babel-generator
andsource-map
(among others) are dependencies ofbabel-core
.babel-core
'spackage.json
lists both these packages as keys independencies
, as shown below.add
Add local or remote
package
as dependency to packages in the current Lerna repo. 在当前的Lerna库中,给packages安装本地或远端的依赖;When run, this command will: 执行该命令时:
package
to each applicable package. Applicable are packages that are notpackage
and are in scopepackage.json
)pacakge
到每个适用的包中,适用于哪些不是pacakge
可是又在范围中的 packages中;lerna add
respects the--ignore
,--scope
and--include-filtered-dependencies
flags (see Flags).lerna add
可以有--ignore
,--scope
and--include-filtered-dependencies
这些标识Examples
babel-generator
can be an internal dependency, whilesource-map
is always an external dependency.babel-generator
in thepackage.json
ofbabel-core
is satisfied bypackages/babel-generator
, passing for an internal dependency.source-map
isnpm install
ed (oryarn
ed) like normal.packages/babel-core/node_modules/babel-generator
symlinks topackages/babel-generator
Notes:
npm install
ed (oryarn
ed) like normal.latest
, do not satisfy semver ranges.Webstorm locks up when circular symlinks are present. To prevent this, add
node_modules
to the list of ignored files and folders inPreferences | Editor | File Types | Ignored files and folders
.publish
Publish packages in the current Lerna project. When run, this command does the following:
Creates a new release of the packages that have been updated. Prompts for a new version. Creates a new git commit/tag in the process of publishing to npm.
More specifically, this command will:
lerna updated
to determine which packages need to be published.version
key inlerna.json
.package.json
of all updated packages to their new versions.Note: to publish scoped packages, you need to add the following to each
package.json
:--exact
When run with this flag,
publish
will specify updated dependencies in updated packages exactly (with no punctuation), instead of as semver compatible (with a^
).For more information, see the package.json dependencies documentation.
--npm-tag [tagname]
When run with this flag,
publish
will publish to npm with the given npm dist-tag (defaults tolatest
).This option can be used to publish a
prerelease
orbeta
version.--canary, -c
When run with this flag,
publish
publishes packages in a more granular way (per commit). Before publishing to npm, it creates the newversion
tag by taking the currentversion
, bumping it to the next minor version, adding the provided meta suffix (defaults toalpha
) and appending the current git sha (ex:1.0.0
becomes1.1.0-alpha.81e3b443
).--conventional-commits
When run with this flag,
publish
will use the Conventional Commits Specification to determine the version bump and generate CHANGELOG--changelog-preset
By default, the changelog preset is set to
angular
. In some cases you might want to change either use a another preset or a custom one.Presets are names of built-in or installable configuration for conventional changelog. Presets may be passed as the full name of the package, or the auto-expanded suffix (e.g.,
angular
is expanded toconventional-changelog-angular
).--git-remote [remote]
When run with this flag,
publish
will push the git changes to the specified remote instead oforigin
.--skip-git
When run with this flag,
publish
will publish to npm without running any of the git commands.--skip-npm
When run with this flag,
publish
will update allpackage.json
package versions and dependency versions, but it will not actually publish the packages to npm.This flag can be combined with
--skip-git
to just update versions and dependencies, without committing, tagging, pushing or publishing.--force-publish [packages]
When run with this flag,
publish
will force publish the specified packages (comma-separated) or all packages using*
.--yes
When run with this flag,
publish
will skip all confirmation prompts. Useful in Continuous integration (CI) to automatically answer the publish confirmation prompt.--cd-version
When run with this flag,
publish
will skip the version selection prompt (in independent mode) and use the next specified semantic version. You must still use the--yes
flag to avoid all prompts. This is useful when build systems need to publish without command prompts. Works in both normal and independent modes.If you have any packages with a prerelease version number (e.g.
2.0.0-beta.3
) and you runlerna publish
with--cd-version
and a non-prerelease version increment (major / minor / patch), it will publish those packages in addition to the packages that have changed since the last release.--preid
When run with this flag,
lerna publish --cd-version
will incrementpremajor
,preminor
,prepatch
, orprerelease
versions using the specified prerelease identifier.--repo-version
When run with this flag,
publish
will skip the version selection prompt and use the specified version. Useful for bypassing the user input prompt if you already know which version to publish.--message, -m [msg]
When run with this flag,
publish
will use the provided message when committing the version updates for publication. Useful for integrating lerna into projects that expect commit messages to adhere to certain guidelines, such as projects which use commitizen and/or semantic-release.If the message contains
%s
, it will be replaced with the new global version version number prefixed with a "v". If the message contains%v
, it will be replaced with the new global version version number without the leading "v". Note that this only applies when using the default "fixed" versioning mode, as there is no "global" version when using--independent
.This can be configured in lerna.json, as well:
--allow-branch [glob]
Lerna allows you to specify a glob or an array of globs in your
lerna.json
that your current branch needs to match to be publishable. You can use this flag to override this setting. If yourlerna.json
contains something like this:and you are not on the branch
master
lerna will prevent you from publishing. To force a publish despite this config, pass the--allow-branch
flag:updated
Check which
packages
have changed since the last release (the last git tag).Lerna determines the last git tag created and runs
git diff --name-only v6.8.1
to get all files changed since that tag. It then returns an array of packages that have an updated file.Note that configuration for the
publish
command also affects theupdated
command. For exampleconfig.publish.ignore
--json
When run with this flag,
updated
will return an array of objects in the following format:clean
Remove the
node_modules
directory from all packages.lerna clean
respects the--ignore
,--scope
, and--yes
flags (see Flags).diff
Diff all packages or a single package since the last release.
ls
List all of the public packages in the current Lerna repo.
lerna ls
respects the--ignore
and--scope
flags (see Flags).--json
When run with this flag,
ls
will return an array of objects in the following format:run
Run an npm script in each package that contains that script. A double-dash (
--
) is necessary to pass dashed arguments to the script execution.lerna run
respects the--concurrency
,--scope
,--ignore
,--stream
, and--parallel
flags (see Flags).exec
Run an arbitrary command in each package. A double-dash (
--
) is necessary to pass dashed flags to the spawned command, but is not necessary when all the arguments are positional.lerna exec
respects the--concurrency
,--scope
,--ignore
,--stream
and--parallel
flags (see Flags).To spawn long-running processes, pass the
--parallel
flag:You may also get the name of the current package through the environment variable
LERNA_PACKAGE_NAME
:You may also run a script located in the root dir, in a complicated dir structure through the environment variable
LERNA_ROOT_PATH
:--bail
This flag signifies whether or not the
exec
command should halt execution upon encountering an error thrown by one of the spawned subprocesses. Its default value istrue
.import
Import the package at
<path-to-external-repository>
, with commit history, intopackages/<directory-name>
. Original commit authors, dates and messages are preserved. Commits are applied to the current branch.This is useful for gathering pre-existing standalone packages into a Lerna repo. Each commit is modified to make changes relative to the package directory. So, for example, the commit that added
package.json
will instead addpackages/<directory-name>/package.json
.link
Symlink together all Lerna
packages
that are dependencies of each other in the current Lerna repo.--force-local
When passed, this flag causes the
link
command to always symlink local dependencies regardless of matching version range.Misc
Lerna will log to a
lerna-debug.log
file (same asnpm-debug.log
) when it encounters an error running a command.Lerna also has support for scoped packages.
Running
lerna
without arguments will show all commands/options.lerna.json
lerna
: the current version of Lerna being used.version
: the current version of the repository.commands.publish.ignore
: an array of globs that won't be included inlerna updated/publish
. Use this to prevent publishing a new version unnecessarily for changes, such as fixing aREADME.md
typo.commands.bootstrap.ignore
: an array of globs that won't be bootstrapped when running thelerna bootstrap
command.commands.bootstrap.scope
: an array of globs that restricts which packages will be bootstrapped when running thelerna bootstrap
command.packages
: Array of globs to use as package locations.Common
devDependencies
Most
devDependencies
can be pulled up to the root of a Lerna repo.This has a few benefits:
Note that
devDependencies
providing "binary" executables that are used by npm scripts still need to be installed directly in each package where they're used.For example the
nsp
dependency is necessary in this case forlerna run nsp
(andnpm run nsp
within the package's directory) to work correctly:Git Hosted Dependencies
Lerna allows target versions of local dependent packages to be written as a git remote url with a
committish
(e.g.,#v1.0.0
or#semver:^1.0.0
) instead of the normal numeric version range. This allows packages to be distributed via git repositories when packages must be private and a private npm registry is not desired.Please note that lerna does not perform the actual splitting of git history into the separate read-only repositories. This is the responsibility of the user. (See this comment for implementation details)
In the example above,
lerna bootstrap
will properly symlinkpkg-2
intopkg-1
.lerna publish
will update the committish (#v1.0.0
) inpkg-1
whenpkg-2
changes.Flags
Options to Lerna can come from configuration (
lerna.json
) or on the command line. Additionally options in config can live at the top level or may be applied to specific commands.Example:
In this case
exampleOption
will be "foo" for all commands exceptinit
, where it will be "bar". In all cases it may be overridden to "baz" on the command-line with--example-option=baz
.--concurrency
How many threads to use when Lerna parallelizes the tasks (defaults to
4
)--scope [glob]
Scopes a command to a subset of packages.
--since [ref]
When executing a script or command, scope the operation to packages that have been updated since the specified
ref
. Ifref
is not specified, it defaults to the latest tag.List the contents of packages that have changed since the latest tag:
Run the tests for all packages that have changed since
master
:List all packages that have changed since
some-branch
:This can be particularly useful when used in CI, if you can obtain the target branch a PR will be going into, because you can use that as the
ref
to the--since
option. This works well for PRs going into master as well as feature branches.--flatten
When importing repositories with merge commits with conflicts, the import command will fail trying to apply all commits. The user can use this flag to ask for import of "flat" history, i.e. with each merge commit as a single change the merge introduced.
--ignore [glob]
Excludes a subset of packages when running a command.
The
ignore
flag, when used with thebootstrap
command, can also be set inlerna.json
under thecommands.bootstrap
key. The command-line flag will take precedence over this option.Example
--ignore-scripts
When used with the
bootstrap
command it won't run any lifecycle scripts in bootstrapped packages.--include-filtered-dependencies
Used in combination with any command that accepts
--scope
(bootstrap
,clean
,ls
,run
,exec
). Ensures that all dependencies (and dev dependencies) of any scoped packages (either through--scope
or--ignore
) are operated on as well.This is useful for situations where you want to "set up" a single package that relies on other packages being set up.
--loglevel [silent|error|warn|success|info|verbose|silly]
What level of logs to report. On failure, all logs are written to lerna-debug.log in the current working directory.
Any logs of a higher level than the setting are shown. The default is "info".
--max-buffer [in-bytes]
Set a max buffer length for each underlying process call. Useful for example when someone wants to import a repo with a larger amount of commits while running
lerna import
. In that case the built-in buffer length might not be sufficient.--no-sort
By default, all tasks execute on packages in topologically sorted order as to respect the dependency relationships of the packages in question. Cycles are broken on a best-effort basis in a way not guaranteed to be consistent across Lerna invocations.
Topological sorting can cause concurrency bottlenecks if there are a small number of packages with many dependents or if some packages take a disproportionately long time to execute. The
--no-sort
option disables sorting, instead executing tasks in an arbitrary order with maximum concurrency.This option can also help if you run multiple "watch" commands. Since
lerna run
will execute commands in topologically sorted order, it can end up waiting for a command before moving on. This will block execution when you run "watch" commands, since they typically never end. An example of a "watch" command is runningbabel
with the--watch
CLI flag.--hoist [glob]
Install external dependencies matching
glob
at the repo root so they're available to all packages. Any binaries from these dependencies will be linked into dependent packagenode_modules/.bin/
directories so they're available for npm scripts. If the option is present but noglob
is given the default is**
(hoist everything). This option only affects thebootstrap
command.For background on
--hoist
, see the hoist documentation.Note: If packages depend on different versions of an external dependency, the most commonly used version will be hoisted, and a warning will be emitted.
--nohoist [glob]
Do not install external dependencies matching
glob
at the repo root. This can be used to opt out of hoisting for certain dependencies.--npm-client [client]
Install external dependencies using
[client] install
. Must be an executable that knows how to install npm dependencies.May also be configured in
lerna.json
:--reject-cycles
Fail immediately if a cycle is found (in
bootstrap
,exec
,publish
orrun
).--use-workspaces
Enables integration with Yarn Workspaces (available since yarn@0.27+). The values in the array are the commands in which Lerna will delegate operation to Yarn (currently only bootstrapping). If
--use-workspaces
is true thenpackages
will be overridden by the value frompackage.json/workspaces.
May also be configured inlerna.json
:The root-level package.json must also include a
workspaces
array:This list is broadly similar to lerna's
packages
config (a list of globs matching directories with a package.json), except it does not support recursive globs ("**"
, a.k.a. "globstars").--stream
Stream output from child processes immediately, prefixed with the originating package name. This allows output from different packages to be interleaved.
--parallel
Similar to
--stream
, but completely disregards concurrency and topological sorting, running a given command or script immediately in all matching packages with prefixed streaming output. This is the preferred flag for long-running processes such asbabel src -d lib -w
run over many packages.--registry [registry]
When run with this flag, forwarded npm commands will use the specified registry for your package(s).
This is useful if you do not want to explicitly set up your registry configuration in all of your package.json files individually when e.g. using private registries.
--temp-tag
When passed, this flag will alter the default publish process by first publishing all changed packages to a temporary dist-tag (
lerna-temp
) and then moving the new version(s) to the default dist-tag (latest
).This is not generally necessary, as Lerna will publish packages in topological order (all dependencies before dependents) by default.
README Badge
Using Lerna? Add a README badge to show it off:
Wizard
If you prefer some guidance for cli (in case you're about to start using lerna or introducing it to a new team), you might like lerna-wizard. It will lead you through a series of well-defined steps: