npm-stage

Stage packages for publishing

Select CLI Version:

Synopsis

npm stage

Note: This command is unaware of workspaces.

Description

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:

  • Staged packages share the same semver version unique index as published packages — you cannot publish a version that already exists as a staged version for that package.
  • You can still publish packages normally while you have staged packages pending.
  • You can stage multiple versions of the same package.
  • npm stage publish has parity with npm publish and will respect "private": true in package.json, refusing to stage the package.

Prerequisites

Before using npm stage commands, ensure the following requirements are met:

  • Write permissions on the package: You must have write access to the package you're configuring.
  • Package must exist: The package you're configuring must already exist on the npm registry.
  • 2FA enabled on your account: Commands that require 2FA will prompt you to authenticate. If you don't already have 2FA enabled on your account, you must enable it before using these commands.

Subcommands

  • npm stage publish [<package-spec>] - Stage a package for publishing
  • npm stage list [<package-spec>] - List all staged package versions
  • npm stage view <stage-id> - View details of a specific staged package
  • npm stage approve <stage-id> - Approve a staged package for publishing
  • npm stage reject <stage-id> - Reject a staged package
  • npm stage download <stage-id> - Download the tarball for inspection

2FA Requirements by Subcommand

CommandRequires 2FANotes
npm stage publishNoDesigned for automated workflows; defers 2FA to approval
npm stage listNoView staged packages
npm stage viewNoView staged package details
npm stage approveYesPrompts for 2FA to publish the staged package
npm stage rejectYesPrompts for 2FA to permanently remove the staged package
npm stage downloadNoDownloads the tarball for local inspection

Tag Behavior

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.

Token Behavior

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 Typenpm stage publishnpm publish
GAT with bypassCan stageCan publish (if allowed by package publishing access)
GAT without bypassCan stage2FA prompt (if allowed by package publishing access)
Session tokenCan stage2FA prompt
Trust token (OIDC)Can stage (if allowed)Can publish (if allowed)

Trust Relationship Permissions

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.

Best Practices

Note: The addition of staged publishing does not make your account or org more secure. Maintainers must still use the best practices listed below.

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

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

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

Configuration

npm stage publish

Stage a package for publishing, deferring proof-of-presence (2FA) to a later point in time

Synopsis

npm stage publish <package-spec>

Flags

FlagDefaultTypeDescription
--tag"latest"StringIf 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 list

List all staged package versions

Synopsis

npm stage list [<package-spec>]

Flags

FlagDefaultTypeDescription
--jsonfalseBooleanWhether 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/"URLThe base URL of the npm registry.

npm stage view

View details of a specific staged package

Synopsis

npm stage view <stage-id>

Flags

FlagDefaultTypeDescription
--jsonfalseBooleanWhether 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/"URLThe base URL of the npm registry.

npm stage approve

Approve a staged package, publishing it to the npm registry

Synopsis

npm stage approve <stage-id>

Flags

FlagDefaultTypeDescription
--otpnullnull or StringThis 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/"URLThe base URL of the npm registry.

npm stage reject

Reject a staged package, removing it from the registry

Synopsis

npm stage reject <stage-id>

Flags

FlagDefaultTypeDescription
--otpnullnull or StringThis 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/"URLThe base URL of the npm registry.

npm stage download

Download the tarball of a staged package for inspection

Synopsis

npm stage download <stage-id>

Flags

FlagDefaultTypeDescription
--jsonfalseBooleanWhether 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/"URLThe base URL of the npm registry.

See Also