Cloudflare is now backing pkg.pr.new's data infrastructure!
pkg.pr.new 
We call it "Continuous Releases" too.
With pkg.pr.new, each of your commits and pull requests will trigger an instant preview release without publishing anything to NPM. This enables users to access features and bug-fixes without the need to wait for release cycles using npm or pull request merges.
- 🚀 Instant Builds
- 🍕 No Need for NPM Access
- 🛠️ GitHub Workflows Friendly
- 📦️ No Configuration
- 🔩 Single Command
- ✉️ Pull Request Comments
- 🔥 Check Runs
pkg.pr.new won't publish anything to NPM; instead, it leverages its own URLs, which are npm-compatible.
npm i https://pkg.pr.new/tinylibs/tinybench/tinybench@a832a55
# npm i https://pkg.pr.new/${owner}/${repo}/${package}@{commit}It is aiming to reduce the number of these comments :)
This was fixed in #18. Can we release that fix?
Badge
Show that your project uses pkg.pr.new by adding a badge to your README:
How to Get a Badge for Your Repository
- Automatically on Repository Page: Visit your repository page at - https://pkg.pr.new/~/OWNER/REPOand you'll find a badge with a copy button that will copy the markdown code for you.
- Direct Badge URL: Use this format to create a badge for your repository: - [](https://pkg.pr.new/~/OWNER/REPO)- Just replace - OWNERand- REPOwith your GitHub username/organization and repository name.
These are some of the projects and companies using pkg.pr.new:
  
  
 
  
  
 
   
  
  
  
 
  
  
  
  
 
   <picture><source media="(prefers-color-scheme: dark)" srcset="https://www.sanity.io/static/images/logo_white.svg"><source media="(prefers-color-scheme: light)" srcset="https://www.sanity.io/static/images/logo_black.svg">
  <picture><source media="(prefers-color-scheme: dark)" srcset="https://www.sanity.io/static/images/logo_white.svg"><source media="(prefers-color-scheme: light)" srcset="https://www.sanity.io/static/images/logo_black.svg"></picture>
  
  
  
  
  
  
 
   
  
  
  
  
    <picture>
        <source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/intlify/art/master/logo_symbol_negative.svg" height="40" />
        
    </picture>
  
  
    <picture>
        <source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/vite-pwa/.github/main/hero-dark.svg" height="40" />
        
    </picture>
  
   
   
   
   
   
   
 
   
   
   
   
   
 
   
   
   
   
   
 
   
   
   
 
    
    
    <picture><source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/preply/design-system-visual-coverage/refs/heads/main/assets/logo-inverted.svg"><source media="(prefers-color-scheme: light)" srcset="https://raw.githubusercontent.com/preply/design-system-visual-coverage/refs/heads/main/assets/logo.svg">
  <picture><source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/preply/design-system-visual-coverage/refs/heads/main/assets/logo-inverted.svg"><source media="(prefers-color-scheme: light)" srcset="https://raw.githubusercontent.com/preply/design-system-visual-coverage/refs/heads/main/assets/logo.svg"></picture>
  
Feel free to add your project or company here to join the pkg.pr.new family :)
You can also join our discord server here, so we all have fun together!
Setup
First install the GitHub Application.
[!IMPORTANT] Make sure it's installed on the repository before trying to publish a package. To read about the permissions the app needs, check #305.
After installing on your repository, you can run npx pkg-pr-new publish in your workflows and then you have continuous releases!
npm install --save-dev pkg-pr-new # or `npx pkg-pr-new publish`For workspaces and monorepos:
npx pkg-pr-new publish './packages/A' './packages/B' # or `npx pkg-pr-new publish './packages/*'`[!IMPORTANT] Make sure the pkg-pr-new command runs only once in a defined workflow (that's part of how it avoids spam)! So instead of multiple times running pkg-pr-new for each workspace package, the command should be run one time for all the desired packages using the pattern above.
For templates (experimental):
[!NOTE] With templates, pkg.pr.new will generate Stackblitz instances for the given directories with the new built packages.
npx pkg-pr-new publish './packages/A' --template './examples/*'By default, pkg.pr.new will generate a template called "default" which includes each built package in the dependencies. This can be disabled with --no-template.
For shorter urls, --compact can be useful:
npx pkg-pr-new publish --compact './packages/A' './packages/B'
--compactrequires your package to be a valid (published) package on npm with a specifiedrepositoryfield in the package.json! See this. pkg.pr.new is case sensitive, if the GitHub owner isPuruVJ, the package.jsonrepositoryfield should not havepuruvj.
With --compact:
npm i https://pkg.pr.new/tinybench@a832a55Without --compact:
npm i https://pkg.pr.new/tinylibs/tinybench/tinybench@a832a55For CLI applications you might want to show npx instead of npm i for the preview command. This can be accomplished with the --bin flag:
npx pkg-pr-new publish --binWith --bin:
npx https://pkg.pr.new/pkg-pr-new@a832a55Without --bin:
npm i https://pkg.pr.new/pkg-pr-new@a832a55You can control publishing comments with --comment:
npx pkg-pr-new publish --comment=update # defaultUsing --comment=update, pkg.pr.new would generate one initial comment and then edit it in the following commits.
With --comment=create, each commit would generate a comment for itself, useful for triggering workflows, like workflow execution using maintainer comments.
And --comment=off would turn off comments for maintainers who prefer minimal pull requests.
To customize which package manager is reflected in the comments, use the --packageManager=XYZ flag. XYZ can be one of the following: npm (default), pnpm, yarn, or bun.
For repositories with many packages, comments might get too long. In that case, you can use --only-templates to only show templates.
pkg.pr.new uses npm pack --json under the hood, in case you face issues, you can also use the --pnpm or --yarn flag so it starts using pnpm pack or yarn pack. This is not necessary in most cases.
pkg.pr.new is not available in your local environment and it only works in workflows.
Examples
Release each commit and pull request:
name: Publish Any Commit
on: [push, pull_request]
jobs:
  build:
    runs-on: ubuntu-latest
    steps:
      - name: Checkout code
        uses: actions/checkout@v4
      - run: corepack enable
      - uses: actions/setup-node@v4
        with:
          node-version: 20
          cache: "pnpm"
      - name: Install dependencies
        run: pnpm install
      - name: Build
        run: pnpm build
      - run: pnpm dlx pkg-pr-new publishRelease approved pull requests only:
name: Publish Approved Pull Requests
on:
  pull_request_review:
    types: [submitted]
jobs:
  check:
    # First, trigger a permissions check on the user approving the pull request.
    if: github.event.review.state == 'approved'
    runs-on: ubuntu-latest
    outputs:
      has-permissions: ${{ steps.checkPermissions.outputs.require-result }}
    steps:
      - name: Check permissions
        id: checkPermissions
        uses: actions-cool/check-user-permission@v2
        with:
          # In this example, the approver must have the write access
          # to the repository to trigger the package preview.
          require: "write"
  publish:
    needs: check
    # Publish the preview package only if the permissions check passed.
    if: needs.check.outputs.has-permissions == 'true'
    runs-on: ubuntu-latest
    steps:
      - name: Checkout code
        uses: actions/checkout@v4
      - run: corepack enable
      - uses: actions/setup-node@v4
        with:
          node-version: 20
          cache: "pnpm"
      - name: Install dependencies
        run: pnpm install
      - run: pnpm dlx pkg-pr-new publishReleasing approved pull requests is the recommended way of having continuous releases. This ensures users always install approved and safe packages.
[!TIP] For any in-repo branch not yet opened as a pull request, if pkg.pr.new has already run on it, a
@branch-nametag will link to the latest commit.
Example: http://pkg.pr.new/vite@main
Avoid publishing on tags
on:
  pull_request:
  push:
    branches:
      - "**"
    tags:
      - "!**"As noted in #140, workflows run on tags too, that's not an issue at all, but in case users would like to avoid duplicate publishes.
Run E2E test using outputs
After pkg-pr-new publish runs successfully, some outputs are available.
- sha: The short SHA used. (E.g.- a832a55)
- urls: Space-separated URLs of published packages.
- packages: Space-separated, Yarn-compatible package locators of published packages.
This is useful for using published packages in other subsequent jobs immediately after publishing. (E.g. E2E tests)
name: Publish and Test Packages
on: [push, pull_request]
jobs:
  publish:
    runs-on: ubuntu-latest
    outputs:
      sha: ${{ steps.publish.outputs.sha }}
      urls: ${{ steps.publish.outputs.urls }}
    steps:
      - name: Checkout code
        uses: actions/checkout@v4
      - run: corepack enable
      - uses: actions/setup-node@v4
        with:
          node-version: 20
          cache: "pnpm"
      - name: Install dependencies
        run: pnpm install
      - name: Build
        run: pnpm build
      - id: publish
        run: pnpm dlx pkg-pr-new publish
  e2e-test:
    runs-on: ubuntu-latest
    needs: publish
    steps:
      - name: Checkout code
        uses: actions/checkout@v4
        with:
          repository: user/my-package-e2e
      - run: corepack enable
      - uses: actions/setup-node@v4
        with:
          node-version: 20
      - name: Install dependencies
        run: pnpm install
      - name: Install published package
        run: pnpm add ${{ needs.publish.outputs.urls }}
      - name: Run e2e test cases
        run: # ...Custom GitHub Messages and Comments
For advanced use cases where you want more control over the messages posted by pkg.pr.new, you can use the --json option in combination with --comment=off. This allows you to generate metadata about the publish operation without creating a default comment, which you can then use to create custom comments via the GitHub Actions API.
Steps:
- Use pkg.pr.new with the --jsonand--comment=offoptions in your workflow:
- name: Publish packages
  run: npx pkg-pr-new publish --json output.json --comment=off- Add a custom step in your workflow to process the JSON output and create a custom comment:
- name: Post or update comment
  uses: actions/github-script@v6
  with:
    github-token: ${{ secrets.GITHUB_TOKEN }}
    script: |
      const fs = require('fs');
      const output = JSON.parse(fs.readFileSync('output.json', 'utf8'));
      const packages = output.packages
        .map((p) => `- ${p.name}: ${p.url}`)
        .join('\n');
      const templates = output.templates
        .map((t) => `- [${t.name}](${t.url})`)
        .join('\n');
      const sha =
        context.event_name === 'pull_request'
          ? context.payload.pull_request.head.sha
          : context.payload.after;
      const commitUrl = `https://github.com/${context.repo.owner}/${context.repo.repo}/commit/${sha}`;
      const body = `## Custom Publish Message
      ### Published Packages:
      ${packages}
      ### Templates:
      ${templates}
      [View Commit](${commitUrl})`;
      const botCommentIdentifier = '## Custom Publish Message';
      async function findBotComment(issueNumber) {
        if (!issueNumber) return null;
        const comments = await github.rest.issues.listComments({
          owner: context.repo.owner,
          repo: context.repo.repo,
          issue_number: issueNumber,
        });
        return comments.data.find((comment) =>
          comment.body.includes(botCommentIdentifier)
        );
      }
      async function createOrUpdateComment(issueNumber) {
        if (!issueNumber) {
          console.log('No issue number provided. Cannot post or update comment.');
          return;
        }
        const existingComment = await findBotComment(issueNumber);
        if (existingComment) {
          await github.rest.issues.updateComment({
            owner: context.repo.owner,
            repo: context.repo.repo,
            comment_id: existingComment.id,
            body: body,
          });
        } else {
          await github.rest.issues.createComment({
            issue_number: issueNumber,
            owner: context.repo.owner,
            repo: context.repo.repo,
            body: body,
          });
        }
      }
      async function logPublishInfo() {
        console.log('\n' + '='.repeat(50));
        console.log('Publish Information');
        console.log('='.repeat(50));
        console.log('\nPublished Packages:');
        console.log(packages);
        console.log('\nTemplates:');
        console.log(templates);
        console.log(`\nCommit URL: ${commitUrl}`);
        console.log('\n' + '='.repeat(50));
      }
      if (context.eventName === 'pull_request') {
        if (context.issue.number) {
          await createOrUpdateComment(context.issue.number);
        }
      } else if (context.eventName === 'push') {
        const pullRequests = await github.rest.pulls.list({
          owner: context.repo.owner,
          repo: context.repo.repo,
          state: 'open',
          head: `${context.repo.owner}:${context.ref.replace(
            'refs/heads/',
            ''
          )}`,
        });
        if (pullRequests.data.length > 0) {
          await createOrUpdateComment(pullRequests.data[0].number);
        } else {
          console.log(
            'No open pull request found for this push. Logging publish information to console:'
          );
          await logPublishInfo();
        }
      }This custom script does the following:
- For pull requests: It creates or updates a comment with the publish information.
- For pushes with an associated open PR: It adds or updates a comment on that PR.
- For pushes without an open PR (e.g., direct pushes to main): It logs the publish information to the GitHub Actions console.
This is a sample recipe that users can adapt with --json and --comment=off to create custom comments.
This custom approach gives you full control over how pkg.pr.new communicates its results.
Publishing is only available in workflows and it supports any workflow trigger event, more information here.
 stackblitz-labs
stackblitz-labs