Select CLI Version:
npm stage
Note: This command is unaware of workspaces.
Staged publishing allows package maintainers to require proof-of-presence for all publishes. Proof-of-presence is where a human is involved, interjects, and provides authentication (2FA) during an action — in this case, publishing an npm package.
Typically when maintainers use automated workflows to publish, proof-of-presence is lacking as there's no convenient way to interject the process and provide 2FA, as is the case for publishing with a granular access token with bypass and the trusted publishing flow. Staged publishing allows users to have their automated workflows stage a package without a 2FA prompt, deferring the act of 2FA, allowing the maintainer to approve the staged package and publish at a later point.
The npm stage publish command packs the current working directory and places that version of the package into the registry in a state where it's not available for public access, allowing maintainers to approve the package at a later point in time. The act of staging does not prompt for 2FA and can be done with any token type, the act of approving will.
Key behaviors:
npm stage publish has parity with npm publish and will respect "private": true in package.json, refusing to stage the package.Before using npm stage commands, ensure the following requirements are met:
npm stage publish [<package-spec>] - Stage a package for publishingnpm stage list [<package-spec>] - List all staged package versionsnpm stage view <stage-id> - View details of a specific staged packagenpm stage approve <stage-id> - Approve a staged package for publishingnpm stage reject <stage-id> - Reject a staged packagenpm stage download <stage-id> - Download the tarball for inspection| Command | Requires 2FA | Notes |
|---|---|---|
npm stage publish | No | Designed for automated workflows; defers 2FA to approval |
npm stage list | No | View staged packages |
npm stage view | No | View staged package details |
npm stage approve | Yes | Prompts for 2FA to publish the staged package |
npm stage reject | Yes | Prompts for 2FA to permanently remove the staged package |
npm stage download | No | Downloads the tarball for local inspection |
The --tag flag follows the same logic as npm publish. If no tag is provided, the latest tag is used by default. For pre-release versions (e.g., 1.0.0-beta.1) and non-latest semver versions, the tag must be explicitly provided — otherwise the CLI will error, just as npm publish would.
The tag is an immutable property of the staged package. Once a package is staged with a given tag, the tag cannot be changed. If you need to stage the same version with a different tag, you must first reject the existing staged package using npm stage reject and then re-stage it with the desired tag.
The key difference with staged publishing is that npm stage publish never requires a 2FA prompt, regardless of token type. This is what makes it suitable for automated workflows. The goal of npm stage publish is deferring proof-of-presence to a later point in time.
| Token Type | npm stage publish | npm publish |
|---|---|---|
| GAT with bypass | Can stage | Can publish (if allowed by package publishing access) |
| GAT without bypass | Can stage | 2FA prompt (if allowed by package publishing access) |
| Session token | Can stage | 2FA prompt |
| Trust token (OIDC) | Can stage (if allowed) | Can publish (if allowed) |
With staged publishing, trust relationships now support granular command permissions. Shortlived tokens issued through trust relationships can only be used with npm stage publish and npm publish. Shortlived tokens cannot run npm stage subcommands.
npm trust <provider> supports --allow-publish and --allow-stage-publish to control which commands are available through each trust relationship.
Note: The addition of staged publishing does not make your account or org more secure. Maintainers must still use the best practices listed below.
Delete Granular Access Tokens (GAT) with bypass 2FA enabled. Now with staged publishing, we've eliminated the need for a GAT token that can bypass 2FA. We encourage you to delete all your tokens with bypass enabled and switch to using a trust relationship in your automated workflows, or create a GAT without bypass and use npm stage publish.
Disallow tokens from publishing at the package level. All packages have their own access controls under "package access" allowing packages to be published with bypass tokens, which is no longer a necessity. We encourage you to select "Require two-factor authentication and disallow tokens (recommended)" for all your packages on the package access page.
Configure trust relationship permissions to prevent npm publish. We encourage you to only enable npm stage publish on your trust relationships and disable npm publish.
npm stage publishStage a package for publishing, deferring proof-of-presence (2FA) to a later point in time
npm stage publish <package-spec>
| Flag | Default | Type | Description |
|---|---|---|---|
--tag | "latest" | String | If you ask npm to install a package and don't tell it a specific version, then it will install the specified tag. It is the tag added to the package@version specified in the npm dist-tag add command, if no explicit tag is given. When used by the npm diff command, this is the tag used to fetch the tarball that will be compared with the local files by default. If used in the npm publish command, this is the tag that will be added to the package submitted to the registry. |
--access |
'public' for new packages, existing packages it will not change the current level
| null, "restricted", or "public" | If you do not want your scoped package to be publicly viewable (and installable) set --access=restricted. Unscoped packages cannot be set to restricted. Note: This defaults to not changing the current access level for existing packages. Specifying a value of restricted or public during publish will change the access for an existing package the same way that npm access set status would. | | --dry-run | false | Boolean | Indicates that you don't want npm to make any changes and that it should only report what it would have done. This can be passed into any of the commands that modify your local installation, eg, install, update, dedupe, uninstall, as well as pack and publish. Note: This is NOT honored by other network related commands, eg dist-tags, owner, etc. | | --otp | null | null or String | This is a one-time password from a two-factor authenticator. It's needed when publishing or changing package permissions with npm access. If not set, and a registry response fails with a challenge for a one-time password, npm will prompt on the command line for one. | | --workspace, -w | | String (can be set multiple times) | Enable running a command in the context of the configured workspaces of the current project while filtering by running only the workspaces defined by this configuration option. Valid values for the workspace config are either: _ Workspace names _ Path to a workspace directory * Path to a parent workspace directory (will result in selecting all workspaces within that folder) When set for the npm init command, this may be set to the folder of a workspace which does not yet exist, to create the folder and set it up as a brand new workspace within the project. | | --workspaces | null | null or Boolean | Set to true to run the command in the context of all configured workspaces. Explicitly setting this to false will cause commands like install to ignore workspaces altogether. When not set explicitly: - Commands that operate on the node_modules tree (install, update, etc.) will link workspaces into the node_modules folder. - Commands that do other things (test, exec, publish, etc.) will operate on the root project, unless one or more workspaces are specified in the workspace config. | | --include-workspace-root | false | Boolean | Include the workspace root when workspaces are enabled for a command. When false, specifying individual workspaces via the workspace config, or all workspaces via the workspaces flag, will cause npm to operate only on the specified workspaces, and not on the root project. | | --provenance | false | Boolean | When publishing from a supported cloud CI/CD system, the package will be publicly linked to where it was built and published from. |
npm stage listList all staged package versions
npm stage list [<package-spec>]
| Flag | Default | Type | Description |
|---|---|---|---|
--json | false | Boolean | Whether or not to output JSON data, rather than the normal output. * In npm pkg set it enables parsing set values with JSON.parse() before saving them to your package.json. Not supported by all npm commands. |
--registry | "https://registry.npmjs.org/" | URL | The base URL of the npm registry. |
npm stage viewView details of a specific staged package
npm stage view <stage-id>
| Flag | Default | Type | Description |
|---|---|---|---|
--json | false | Boolean | Whether or not to output JSON data, rather than the normal output. * In npm pkg set it enables parsing set values with JSON.parse() before saving them to your package.json. Not supported by all npm commands. |
--registry | "https://registry.npmjs.org/" | URL | The base URL of the npm registry. |
npm stage approveApprove a staged package, publishing it to the npm registry
npm stage approve <stage-id>
| Flag | Default | Type | Description |
|---|---|---|---|
--otp | null | null or String | This is a one-time password from a two-factor authenticator. It's needed when publishing or changing package permissions with npm access. If not set, and a registry response fails with a challenge for a one-time password, npm will prompt on the command line for one. |
--registry | "https://registry.npmjs.org/" | URL | The base URL of the npm registry. |
npm stage rejectReject a staged package, removing it from the registry
npm stage reject <stage-id>
| Flag | Default | Type | Description |
|---|---|---|---|
--otp | null | null or String | This is a one-time password from a two-factor authenticator. It's needed when publishing or changing package permissions with npm access. If not set, and a registry response fails with a challenge for a one-time password, npm will prompt on the command line for one. |
--registry | "https://registry.npmjs.org/" | URL | The base URL of the npm registry. |
npm stage downloadDownload the tarball of a staged package for inspection
npm stage download <stage-id>
| Flag | Default | Type | Description |
|---|---|---|---|
--json | false | Boolean | Whether or not to output JSON data, rather than the normal output. * In npm pkg set it enables parsing set values with JSON.parse() before saving them to your package.json. Not supported by all npm commands. |
--registry | "https://registry.npmjs.org/" | URL | The base URL of the npm registry. |