npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2026 – Pkg Stats / Ryan Hefner

workspaces-publish

v1.8.1

Published

A library to publish packages of multiple workspaces.

Readme

workspaces-publish

npm version Master Workflow Known Vulnerabilities Conventional Commits

This library facilitates the publication of packages encompassing multiple workspaces as defined in a package.json file. It determines the unpublished packages by checking each package manifest of the registry, if one already exists.

At best, it should be used with a library that increments the version of the packages beforehand (e.g. release-please).

When npm >= 10.0.0 is available, it shells out to npm publish directly (supporting OIDC, provenance, etc. out of the box). Otherwise it falls back to libnpmpublish / libnpmpack.

Table of Contents

Installation

npm install workspaces-publish --save-dev

Usage

npx workspaces-publish \
  --token <token> \
  --registry <registry> \
  --root <root> \
  --rootPackage

Options

token

  • Type: String
  • Default: process.env.NODE_AUTH_TOKEN
  • Description: Token for the registry. Optional when using OIDC trusted publishing.

registry

  • Type: String
  • Default: https://registry.npmjs.org/
  • Description: Registry url.

root

  • Type: String
  • Default: process.cwd()
  • Description: Directory where the root package is located.

rootPackage

  • Type: Boolean
  • Default: true
  • Description: Also consider the root package for publishing. The library still checks whether a name- & version-property is set and whether the private property evaluates to false.

Authentication

The library supports three authentication methods, resolved in the following order:

  1. --token CLI flag — Explicit npm access token
  2. OIDC Trusted Publishing — Tokenless publishing via GitHub Actions OIDC (auto-detected)
  3. NODE_AUTH_TOKEN environment variable — Fallback to the environment variable

OIDC Trusted Publishing

When running in GitHub Actions with trusted publishers configured, the library automatically detects the OIDC environment and exchanges short-lived tokens with the npm registry — no long-lived NPM_TOKEN required.

Requirements:

  • npm trusted publisher configured for each package on npmjs.com
  • GitHub Actions workflow with id-token: write permission
  • Node.js >= 22.0.0
  • No --token flag or NODE_AUTH_TOKEN set (OIDC activates only when no explicit token is present)

How it works:

  1. Detects ACTIONS_ID_TOKEN_REQUEST_URL and ACTIONS_ID_TOKEN_REQUEST_TOKEN environment variables
  2. Requests an OIDC identity token from GitHub with audience npm:<registry-host>
  3. Exchanges the identity token with the npm registry for a short-lived, package-scoped publish token
  4. Uses that token for publishing (each package gets its own token)

Programmatic API

import { publish } from 'workspaces-publish';

const packages = await publish({
    cwd: '/path/to/monorepo',
    registry: 'https://registry.npmjs.org/',
    token: 'npm_...',
    rootPackage: true,
    dryRun: false,
});

Custom Adapters

The library uses a hexagonal architecture (ports & adapters). All external I/O is abstracted behind interfaces, making it fully testable and extensible:

import { publish } from 'workspaces-publish';
import {
    MemoryFileSystem,
    MemoryRegistryClient,
    MemoryPublisher,
    MemoryTokenProvider,
    NoopLogger,
} from 'workspaces-publish';

// Use memory adapters for testing
const packages = await publish({
    cwd: '/project',
    fileSystem: new MemoryFileSystem({ /* virtual files */ }),
    registryClient: new MemoryRegistryClient({ /* virtual packuments */ }),
    publisher: new MemoryPublisher(),
    tokenProvider: new MemoryTokenProvider('test-token'),
    logger: new NoopLogger(),
});

Available port interfaces and their adapters:

| Port | Real Adapter | Test Adapter | |------|-------------|-------------| | IFileSystem | NodeFileSystem | MemoryFileSystem | | IRegistryClient | HapicRegistryClient | MemoryRegistryClient | | IPackagePublisher | NpmCliPublisher (npm CLI), NpmPublisher (libnpmpublish fallback) | MemoryPublisher | | ITokenProvider | StaticTokenProvider, EnvTokenProvider, OidcTokenProvider, ChainTokenProvider | MemoryTokenProvider | | ILogger | ConsolaLogger | NoopLogger |

CI

GitHub Action (with npm token)

The library can be used in combination with release-please, as release-please only increases the versions in the monorepo, but does not release the packages.

on:
    push:
        branches:
            - main

permissions:
    contents: write
    pull-requests: write

jobs:
    release:
        runs-on: ubuntu-latest
        steps:
            -   uses: google-github-actions/release-please-action@v4
                id: release
                with:
                    token: ${{ secrets.GITHUB_TOKEN }}

            -   name: Checkout
                if: steps.release.outputs.releases_created == 'true'
                uses: actions/checkout@v4

            -   name: Install Node.JS
                if: steps.release.outputs.releases_created == 'true'
                uses: actions/setup-node@v4
                with:
                    node-version: 22

            -   name: Install dependencies
                if: steps.release.outputs.releases_created == 'true'
                run: npm ci

            -   name: Publish
                if: steps.release.outputs.releases_created == 'true'
                run: npx workspaces-publish
                env:
                    NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }}

GitHub Action (with OIDC Trusted Publishing)

No npm token secrets needed — configure trusted publishers on npmjs.com for each package instead.

on:
    push:
        branches:
            - main

permissions:
    contents: write
    pull-requests: write
    id-token: write

jobs:
    release:
        runs-on: ubuntu-latest
        steps:
            -   uses: google-github-actions/release-please-action@v4
                id: release
                with:
                    token: ${{ secrets.GITHUB_TOKEN }}

            -   name: Checkout
                if: steps.release.outputs.releases_created == 'true'
                uses: actions/checkout@v4

            -   name: Install Node.JS
                if: steps.release.outputs.releases_created == 'true'
                uses: actions/setup-node@v4
                with:
                    node-version: 22

            -   name: Install dependencies
                if: steps.release.outputs.releases_created == 'true'
                run: npm ci

            -   name: Publish
                if: steps.release.outputs.releases_created == 'true'
                run: npx workspaces-publish