Skip to content

Latest commit

 

History

History
155 lines (99 loc) · 10 KB

migrating-an-npm-package-to-the-liferay-named-scope.md

File metadata and controls

155 lines (99 loc) · 10 KB

Migrating an npm package to the @liferay named scope

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:

  1. Migrate the project into the monorepo, if it is not already there.
  2. Update all internal references within the project to reflect the new name.
  3. Produce a release of the package under the new name (while preserving the version number).
  4. Configure package access control so that team members can publish.
  5. Add a deprecation notice to the old package, directing consumers to access the package under the new name instead.
  6. Update labels and any other repo "meta" artifacts that contained the old name.

Details

Migrating the project into the monorepo

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.

Updating all internal references within the project to reflect the new name

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 the package.json (eg. eslint-config-liferay@liferay/eslint-config). Likewise, the repository field should be updated to match the renamed directory.

  • Update the version-tag-prefix in the .yarnrc to drop the liferay 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).

Releasing the package under the new name

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.

  1. Log in as the liferay npm user; using npm login (yarn login won't work, even though you will perform the publish using yarn).

  2. 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.

  3. Tag the release manually with git tag. The tag format should match the version-tag-prefix specified in the .yarnrc file and the message should match the version-git-message. So, for the example of @liferay/eslint-config, that means git tag eslint-config/v21.1.0 -m 'chore: prepare @liferay/eslint-config v21.1.0 release'.

  4. 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
  5. 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

  6. 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) to eslint-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
    

Configuring package access control

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 the liferay-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.

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.

Deprecating the old package

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:

This package has been deprecated


Author message:

Please see @liferay/link-checker instead

Update labels and any other repo "meta" artifacts

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 simply npm-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.

Troubleshooting

You must sign up for private packages

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.

Forbidden

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.