It is desirable to have our packages under the @liferay
named scope for a few reasons:
- Easier discovery (eg. at-a-glance view of all of our packages).
- Easier management (eg. centralized access control).
- Clear way of distinguishing packages as officially "built-by-Liferay" as opposed to just "has-a-liferay-prefix-but-could-be-built-by-anyone".
- Namespace reserved just for us, so we can always secure the package names we want without having to compete with pre-existing registrations.
Moving an existing package to the @liferay
named scope consists of the following steps:
- Migrate the project into the monorepo, if it is not already there.
- Update all internal references within the project to reflect the new name.
- Produce a release of the package under the new name (while preserving the version number).
- Configure package access control so that team members can publish.
- Add a deprecation notice to the old package, directing consumers to access the package under the new name instead.
- Update labels and any other repo "meta" artifacts that contained the old name.
When a project renames itself from liferay-foo
to @liferay/foo
, it is still the same project, despite the different name. It's important that somebody examining this new @liferay/foo
package can use git blame
and git log
to discover the full history of the project without having to jump between repos.
Therefore, if the project isn't already in the monorepo, you should bring it in using a history-preserving subtree merge as described in "Importing a project". Only then should you proceed with the following steps.
A comprehensive example can be seen in this commit which migrates the eslint-config-liferay project to @liferay/eslint-config
. This one is complicated by the conventions that ESLint imposes for named scopes; other migrations should be simpler than this one.
Noteworthy:
-
Rename the project directory to omit the
liferay-
prefix (or suffix). For example:git mv projects/eslint-config-liferay projects/eslint-config
-
Update the
name
field in thepackage.json
(eg.eslint-config-liferay
→@liferay/eslint-config
). Likewise, therepository
field should be updated to match the renamed directory. -
Update the
version-tag-prefix
in the.yarnrc
to drop theliferay
prefix (or suffix). -
Update internal references in code and documentation. Note that links to issues and PRs should be left as-is, because those will continue to function even when the old repository is archived (ie. marked as read-only).
At this point, the package has a new name and there have been no code changes (save those that may be necessary to keep CI passing and functionality intact with the new name). We want to demonstrate the continuity by publishing a release of the package in its new, named-scope home, using the same version number as the previous, unscoped release.
In the example of eslint-config-liferay, the last release was v21.1.0. So we released the corresponding @liferay/eslint-config
v21.1.0.
-
Log in as the
liferay
npm user; usingnpm login
(yarn login
won't work, even though you will perform the publish usingyarn
). -
Run
yarn publish --access=public
to actually make the release. The--access=public
switch is required only the first time you publish a particular package under a named scope. -
Tag the release manually with
git tag
. The tag format should match theversion-tag-prefix
specified in the.yarnrc
file and the message should match theversion-git-message
. So, for the example of@liferay/eslint-config
, that meansgit tag eslint-config/v21.1.0 -m 'chore: prepare @liferay/eslint-config v21.1.0 release'
. -
Push the commits (if any) and tags with a command like this (assuming your upstream remote is called
upstream
):git push upstream --follow-tags --dry-run # sanity-check first git push upstream --follow-tags
-
Check the package page on npmjs.org to confirm the release is showing up correctly; the release should be at a URL like: npmjs.com/package/@liferay/eslint-config
-
Check the release page on GitHub to confirm that the release is showing up there too; it should be at a URL like: github.com/liferay/liferay-frontend-projects/releases/tag/eslint-config/v21.1.0
🚧
@liferay/changelog-generator
can produce release notes for this kind of release but it is rather edge-casey so they may need some manual tweaking. For example, it assumes that the tag prefix doesn't change from release to release like it does here (and we probably won't bother changing it because importing a package is such a once-off thing), so you'll have to manually edit the "compare" URL to show the right starting tag. Other than that, it should work fine.An example invocation, used to document the move from
eslint-config-liferay/v21.1.10
(the tag pointing at the final release of the old package) toeslint-config/v21.1.0
(the first release of the new named-scope version of the package) is:cd projects/eslint-config node ../npm-tools/packages/changelog-generator/bin/liferay-changelog-generator.js --from=eslint-config-liferay/v21.1.0 --version=21.1.0 # Edit the starting tag from "eslint-config/v21.1.0" to "liferay-eslint-config/v21.1.0": vim CHANGELOG.md
The yarn publish
will have created a new package with the liferay
user as the sole owner. You can verify this with yarn owner list
. We need to adapt this to account for our team structure and allow all team members to publish.
The structure is as follows:
liferay
is an npm user.- The
liferay
user belongs to theliferay-frontend
npm organization.- The
liferay-frontend
organization has two teams, of which one is of principal interest to us:- The
developers
team is the default, and contains the bulk of our packages as well as all of the Frontend Infrastructure team.
- The
- The
- The
So, to grant access to the team members it suffices to go to the settings page when logged in as the liferay
user and click the Add package
button.
Once you've done that, yarn owner list
should show that all team members have access.
As a final step, use npm deprecate
to direct people to use the new package. For example:
npm deprecate eslint-config-liferay 'Please see @liferay/eslint-config instead'
Note that npm deprecate
can also take a version range; we used this, for example, with the move of liferay-amd-loader
to @liferay/amd-loader
, where we expect the v3.x series to continue to be used (optionally) on DXP 7.1 for some time:
npm deprecate liferay-amd-loader@">= 4.0.0" 'Please see @liferay/amd-loader instead'
The version range causes a deprecation warning to be shown only when installing v4.0.0 and above:
$ yarn add [email protected]
success Saved 2 new dependencies.
$ yarn add liferay-amd-loader
warning [email protected]: Please see @liferay/amd-loader instead
success Saved 1 new dependency.
Likewise, the package on npmjs.com (example page) will show a message like:
Author message:
Please see @liferay/link-checker instead
If there were any labels or other repo entities that contained the old name, they should be updated too. For example:
- A label like
liferay-npm-scripts
would become simplynpm-scripts
, and should be edited both in the GitHub UI and in the auto-labeler configuration file. - Any top-level configuration files —
.editorconfig
,.eslintignore
,.eslintrc.js
,.gitignore
,.prettierignore
,.prettierrc.js
,jest.config.js
, all files under.github/
etc — should be double-checked to make sure that they don't contain stale references to the old path. Technically, this should already have happened earlier, during "step 2", but it is a good idea to double-check.
If you see output like:
error Couldn't publish package: "https://registry.yarnpkg.com/@liferay%2fsome-package: You must sign up for private packages"
it may be because you didn't supply the --access=public
flag to yarn publish
(necessary only on the first publish) or because the package.json
does not include a "private": false
property.
Output like:
error Couldn't publish package: "https://registry.yarnpkg.com/@liferay%2fsome-package: Forbidden"
it may be because you are not logged in as the liferay
npm user. This is necessary only on the first publish in order to publish under the @liferay
named scope. Once the package is published and access granted to the liferay-frontend team members, they can publish updates logged in to their personal accounts.
Note that logging in with yarn login
may not be sufficient. In my testing, I had to log in with npm login
(even though I subsequently did the publish with yarn publish --access=public
.