7.6 KiB
Release Automation Setup
This document covers the GitHub and npm setup required for the current Paperclip release model:
- automatic canaries from
master - manual stable promotion from a chosen source ref
- npm trusted publishing via GitHub OIDC
- protected release infrastructure in a public repository
Repo-side files that depend on this setup:
.github/workflows/release.yml.github/CODEOWNERS
Note:
- the release workflows intentionally use
pnpm install --no-frozen-lockfile - this matches the repo's current policy where
pnpm-lock.yamlis refreshed by GitHub automation after manifest changes land onmaster - the publish jobs then restore
pnpm-lock.yamlbefore runningscripts/release.sh, so the release script still sees a clean worktree
1. Merge the Repo Changes First
Before touching GitHub or npm settings, merge the release automation code so the referenced workflow filenames already exist on the default branch.
Required files:
.github/workflows/release.yml.github/CODEOWNERS
2. Configure npm Trusted Publishing
Do this for every public package that Paperclip publishes.
At minimum that includes:
paperclipai@paperclipai/server- public packages under
packages/
2.1. In npm, open each package settings page
For each package:
- open npm as an owner of the package
- go to the package settings / publishing access area
- add a trusted publisher for the GitHub repository
paperclipai/paperclip
2.2. Add one trusted publisher entry per package
npm currently allows one trusted publisher configuration per package.
Configure:
- workflow:
.github/workflows/release.yml
Repository:
paperclipai/paperclip
Environment name:
- leave the npm trusted-publisher environment field blank
Why:
- the single
release.ymlworkflow handles both canary and stable publishing - GitHub environments
npm-canaryandnpm-stablestill enforce different approval rules on the GitHub side
2.3. Verify trusted publishing before removing old auth
After the workflows are live:
- run a canary publish
- confirm npm publish succeeds without any
NPM_TOKEN - run a stable dry-run
- run one real stable publish
Only after that should you remove old token-based access.
3. Remove Legacy npm Tokens
After trusted publishing works:
- revoke any repository or organization
NPM_TOKENsecrets used for publish - revoke any personal automation token that used to publish Paperclip
- if npm offers a package-level setting to restrict publishing to trusted publishers, enable it
Goal:
- no long-lived npm publishing token should remain in GitHub Actions
4. Create GitHub Environments
Create two environments in the GitHub repository:
npm-canarynpm-stable
Path:
- GitHub repository
SettingsEnvironmentsNew environment
5. Configure npm-canary
Recommended settings for npm-canary:
- environment name:
npm-canary - required reviewers: none
- wait timer: none
- deployment branches and tags:
- selected branches only
- allow
master
Reasoning:
- every push to
mastershould be able to publish a canary automatically - no human approval should be required for canaries
6. Configure npm-stable
Recommended settings for npm-stable:
- environment name:
npm-stable - required reviewers: at least one maintainer other than the person triggering the workflow when possible
- prevent self-review: enabled
- admin bypass: disabled if your team can tolerate it
- wait timer: optional
- deployment branches and tags:
- selected branches only
- allow
master
Reasoning:
- stable publishes should require an explicit human approval gate
- the workflow is manual, but the environment should still be the real control point
7. Protect master
Open the branch protection settings for master.
Recommended rules:
- require pull requests before merging
- require status checks to pass before merging
- require review from code owners
- dismiss stale approvals when new commits are pushed
- restrict who can push directly to
master
At minimum, make sure workflow and release script changes cannot land without review.
8. Enforce CODEOWNERS Review
This repo now includes .github/CODEOWNERS, but GitHub only enforces it if branch protection requires code owner reviews.
In branch protection for master, enable:
Require review from Code Owners
Then verify the owner entries are correct for your actual maintainer set.
Current file:
.github/CODEOWNERS
If @dotta is not the right reviewer identity in the public repo, change it before enabling enforcement.
9. Protect Release Infrastructure Specifically
These files should always trigger code owner review:
.github/workflows/release.ymlscripts/release.shscripts/release-lib.shscripts/release-package-map.mjsscripts/create-github-release.shscripts/rollback-latest.shdoc/RELEASING.mddoc/PUBLISHING.md
If you want stronger controls, add a repository ruleset that explicitly blocks direct pushes to:
.github/workflows/**scripts/release*
10. Do Not Store a Claude Token in GitHub Actions
Do not add a personal Claude or Anthropic token for automatic changelog generation.
Recommended policy:
- stable changelog generation happens locally from a trusted maintainer machine
- canaries never generate changelogs
This keeps LLM spending intentional and avoids a high-value token sitting in Actions.
11. Verify the Canary Workflow
After setup:
- merge a harmless commit to
master - open the
Releaseworkflow run triggered by that push - confirm it passes verification
- confirm publish succeeds under the
npm-canaryenvironment - confirm npm now shows a new
canaryrelease - confirm a git tag named
canary/vYYYY.M.D-canary.Nwas pushed
Install-path check:
npx paperclipai@canary onboard
12. Verify the Stable Workflow
After at least one good canary exists:
- prepare
releases/vYYYY.M.D.mdon the source commit you want to promote - open
Actions->Release - run it with:
source_ref: the tested commit SHA or canary tag source commitstable_date: leave blank or set the intended UTC datedry_run:true
- confirm the dry-run succeeds
- rerun with
dry_run: false - approve the
npm-stableenvironment when prompted - confirm npm
latestpoints to the new stable version - confirm git tag
vYYYY.M.Dexists - confirm the GitHub Release was created
13. Suggested Maintainer Policy
Use this policy going forward:
- canaries are automatic and cheap
- stables are manual and approved
- only stables get public notes and announcements
- release notes are committed before stable publish
- rollback uses
npm dist-tag, not unpublish
14. Troubleshooting
Trusted publishing fails with an auth error
Check:
- the workflow filename on GitHub exactly matches the filename configured in npm
- the package has the trusted publisher entry for the correct repository
- the job has
id-token: write - the job is running from the expected repository, not a fork
Stable workflow runs but never asks for approval
Check:
- the
publishjob uses environmentnpm-stable - the environment actually has required reviewers configured
- the workflow is running in the canonical repository, not a fork
CODEOWNERS does not trigger
Check:
.github/CODEOWNERSis on the default branch- branch protection on
masterrequires code owner review - the owner identities in the file are valid reviewers with repository access