openclaw-release-maintainer
Installation
SKILL.md
OpenClaw Release Maintainer
Use this skill for release and publish-time workflow. Keep ordinary development changes and GHSA-specific advisory work outside this skill.
Respect release guardrails
- Do not change version numbers without explicit operator approval.
- Ask permission before any npm publish or release step.
- This skill should be sufficient to drive the normal release flow end-to-end.
- Use the private maintainer release docs for credentials, recovery steps, and mac signing/notary specifics, and use
docs/reference/RELEASING.mdfor public policy. - Core
openclawpublish is manualworkflow_dispatch; creating or pushing a tag does not publish by itself. - Normal release work happens on a branch cut from
main, not directly onmain. Userelease/YYYY.M.Dfor the branch name. - If the operator asks for a release without saying stable/full, default to beta only. Continue from beta to stable only when the operator explicitly asks for the full release or an automated beta-and-stable train.
- Before release branching, pull latest
mainand confirm currentmainCI is green. Then branch from that commit so regular development can continue onmainwhile release validation runs. - Before release branching, commit any dirty files in coherent groups, push,
pull/rebase, then run
/changelogonmainand commit/push/pull that changelog rewrite immediately before creating the release branch. - Do not delete or rewrite beta tags after they leave the machine. If a
published or pushed beta needs a fix, commit the fix on the release branch and
increment to the next
-beta.N. - For a beta release train, run the full pre-npm test roster before publishing each beta. After a beta is published, run the smaller published-install roster focused on install/update/Docker/Parallels. If anything fails, fix it on the release branch, commit/push/pull, increment beta number, and repeat. Operators may authorize up to 4 autonomous beta attempts; after 4 failed beta attempts, stop and report.
- Use
/changelogbefore version/tag preparation so the top changelog section is deduped and ordered by user impact. - Do not create beta-specific
CHANGELOG.mdheadings. Beta releases use the stable base version section, for examplev2026.4.20-beta.1uses## 2026.4.20release notes. - When any beta or stable release is live, make a best-effort Discord
announcement using Peter's bot token from
.profile; do not block or roll back the release if the announcement fails. - When asked to announce on X, use
~/Projects/bird/birdand follow the release tweet style below.
Keep release channel naming aligned
stable: tagged releases only, published to npmbetaby default; operators may target npmlatestexplicitly or promote laterbeta: prerelease tags likevYYYY.M.D-beta.N, with npm dist-tagbeta- Prefer
-beta.N; do not mint new-1or-2beta suffixes dev: moving head onmain- When using a beta Git tag, publish npm with the matching beta version suffix so the plain version is not consumed or blocked
Handle versions and release files consistently
- Version locations include:
package.jsonapps/android/app/build.gradle.ktsapps/ios/Sources/Info.plistapps/ios/Tests/Info.plistapps/macos/Sources/OpenClaw/Resources/Info.plistdocs/install/updating.md- Peekaboo Xcode project and plist version fields
- Before creating a release tag, make every version location above match the version encoded by that tag.
- For fallback correction tags like
vYYYY.M.D-N, the repo version locations still stay atYYYY.M.D. - “Bump version everywhere” means all version locations above except
appcast.xml. - Release signing and notary credentials live outside the repo in the private maintainer docs.
- Every stable OpenClaw release ships the npm package and macOS app together. Beta releases normally ship npm/package artifacts first and skip mac app build/sign/notarize unless the operator requests mac beta validation.
- Do not let the slower macOS signing/notary path block npm publication once the npm preflight has passed. Keep mac validation/publish running in parallel, publish npm from the successful npm preflight, then start published npm install/update, Docker, and Parallels verification while mac artifacts continue.
- Mac packaging may be built from a slight release-branch variation of the
tagged commit when the delta is mac packaging, signing, workflow, or
validation-only release machinery. If mac packaging needs release-branch-only
fixes after the stable npm package or GitHub tag is already published, do not
create a
vYYYY.M.D-Ncorrection tag just to change the workflow source. Dispatch the private mac workflows for the originaltag=vYYYY.M.Dwithsource_ref=release/YYYY.M.Dandpublic_release_branch=release/YYYY.M.D; provenance checks must prove the source SHA descends from the tag and validation/preflight use the same source. ReservevYYYY.M.D-Ncorrection tags for emergency hotfixes that must publish a new npm package/release identity, not for ordinary mac-only packaging recovery. - The production Sparkle feed lives at
https://raw.githubusercontent.com/openclaw/openclaw/main/appcast.xml, and the canonical published file isappcast.xmlonmainin theopenclawrepo. - That shared production Sparkle feed is stable-only. Beta mac releases may
upload assets to the GitHub prerelease, but they must not replace the shared
appcast.xmlunless a separate beta feed exists. - For fallback correction tags like
vYYYY.M.D-N, the repo version still stays atYYYY.M.D, but the mac release must use a strictly higher numericAPP_BUILD/ Sparkle build than the original release so existing installs see it as newer.
Build changelog-backed release notes
- Before release branching or tagging, rewrite the target
CHANGELOG.mdsection from commit history, not just from existing notes: scan commits since the last reachable release tag, add missed user-facing changes, dedupe overlapping entries, and sort each section from most to least interesting for users. - Changelog entries should be user-facing, not internal release-process notes.
- GitHub release and prerelease bodies must use the full matching
CHANGELOG.mdversion section, not highlights or an excerpt. When creating or editing a release, extract from## YYYY.M.Dthrough the line before the next level-2 heading and use that complete block as the release notes. - When cutting a mac release with a beta GitHub prerelease:
- tag
vYYYY.M.D-beta.Nfrom the release commit - create a prerelease titled
openclaw YYYY.M.D-beta.N - use release notes from the stable base
CHANGELOG.mdversion section (## YYYY.M.D), not a beta-specific heading - attach at least the zip and dSYM zip, plus dmg if available
- tag
- Keep the top version entries in
CHANGELOG.mdsorted by impact:### Changesfirst### Fixesdeduped with user-facing fixes first
Write release tweets
Use the OpenClaw account's existing release-post style:
- Format:
OpenClaw YYYY.M.D 🦞or🦞 OpenClaw YYYY.M.D is live, blank line, then 3-4 emoji-led bullets, blank line, one short punchline, then the release link. - For beta: say
OpenClaw YYYY.M.D-beta.N 🦞orOpenClaw YYYY.M.D beta N is live; keep it clearly beta and avoid implying stable promotion. - Lead with user-visible capabilities, then important integrations, then reliability/security/install fixes. Compress "lots of fixes" into one readable bullet.
- Read the full changelog section before drafting. Do not lead with coverage, CI, validation, or internal release mechanics unless the release is explicitly about those. Peter prefers concrete user wins: features, integrations, workflow improvements, and practical reliability fixes.
- Tone: high-signal, slightly cheeky, confident, not corporate. One joke is enough. Avoid punching down, insulting users, or promising what was not verified.
- Peter likes dry, compact taglines when they feel earned. Good example:
Big release, tiny release notes... kidding.Keep the joke short and let the feature bullets carry the tweet; do not turn the punchline into a second paragraph or a forced bit. - Length: release tweets are always standard tweets under 280 characters, with room for one URL. Trim to 3-4 bullets and count the final text before posting.
- Links/media: include the GitHub release or changelog link at the end of the first release tweet.
- Thread follow-ups: if doing a thread, keep the first release tweet as the compact launch post, then publish one focused feature explainer per reply. Follow-up replies should not repeat "new in VERSION" or the version number when the thread context already makes it obvious.
- Every follow-up tweet should include a docs URL for that specific feature.
Prefer a bare URL over
Docs: <url>unless the label is needed for clarity. Keep follow-ups concise: around 160-220 raw characters is usually the sweet spot; under 280 is the hard cap. If a URL makes a tweet fail, trim prose before dropping the URL. Prefer explaining diagnostics, trajectory/export, provider setup, model commands, or other setup-heavy features in follow-ups instead of overloading the first release tweet. - Hotfix/correction: be direct and accountable. State what slipped, what is fixed, and the new version. Keep jokes out of incident-style posts.
Examples to adapt:
OpenClaw 2026.4.20-beta.1 🦞
🐳 Docker install/update smoke
🖥️ Parallels upgrade checks
🔧 Package verification tightened
Beta first. Stable after the gauntlet.
<release link>
OpenClaw 2026.4.20 🦞
🚀 Faster install + update
🐳 Docker + Parallels verified
🍎 macOS signed + notarized
🔧 Channel/plugin fixes
Good boring release. Best kind.
<release link>
Packaging issue in 2026.4.20-beta.1.
2026.4.20-beta.2 fixes install/update verification. No tag rewrites; beta moves
forward.
Upgrade with the beta channel.
<release link>
Run publish-time validation
Before tagging or publishing, run:
pnpm check:architecture
pnpm build
pnpm ui:build
pnpm release:check
pnpm test:install:smoke
For a non-root smoke path:
OPENCLAW_INSTALL_SMOKE_SKIP_NONROOT=1 pnpm test:install:smoke
After npm publish, run:
node --import tsx scripts/openclaw-npm-postpublish-verify.ts <published-version>
- This verifies the published registry install path in a fresh temp prefix.
- For stable correction releases like
YYYY.M.D-N, it also verifies the upgrade path fromYYYY.M.DtoYYYY.M.D-Nso a correction publish cannot silently leave existing global installs on the old base stable payload. - Treat install smoke as a pack-budget gate too.
pnpm test:install:smokenow fails the candidate update tarball when npm reports an oversizedunpackedSize, so release-time e2e cannot miss pack bloat that would risk low-memory install/startup failures. - Keep direct npm global coverage enabled in install smoke. It exercises plain
npm install -g <candidate>fresh installs and npm-driven update installs, because many users install with npm even when docs prefer pnpm. - Use
pnpm test:live:media videofor bounded video-provider smoke when video generation is in release scope. The default video smoke skipsfal, runs one text-to-video attempt per provider with a one-second lobster prompt, and caps each provider operation withOPENCLAW_LIVE_VIDEO_GENERATION_TIMEOUT_MS(180000by default). - Run
pnpm test:live:media video --video-providers falonly when FAL-specific proof is required. Its queue latency can dominate release time. - Set
OPENCLAW_LIVE_VIDEO_GENERATION_FULL_MODES=1only when intentionally validating the slower image-to-video and video-to-video transform lanes.
Check all relevant release builds
- Always validate the OpenClaw npm release path before creating the tag.
- Source Peter's profile before live release validation so OpenAI and Anthropic
credentials are available without printing secrets:
set -a; source "$HOME/.profile"; set +a. - Parallels validation and any local live model QA for this train must use both
OPENAI_API_KEYandANTHROPIC_API_KEY. If either is missing after sourcing.profile, stop before starting those local long lanes and report the missing key. - Live credentialed channel QA is the GitHub Actions workflow
QA-Lab - All Lanes(.github/workflows/qa-live-telegram-convex.yml), not a local substitute. Dispatch it from Actions against the release tag and wait for it to pass before npm preflight/publish readiness. Use a SHA only when it satisfies the workflow's secret-bearing trust gate: main ancestor or open PR head. It runs the QA Lab mock parity gate plus live Matrix and live Telegram lanes using theqa-live-sharedenvironment; Telegram uses Convex CI credential leases. - Default release checks:
pnpm checkpnpm check:test-typespnpm check:architecturepnpm buildpnpm ui:buildpnpm release:checkOPENCLAW_INSTALL_SMOKE_SKIP_NONROOT=1 pnpm test:install:smoke
- Full pre-npm beta test roster:
- default release checks above
- all Docker tests:
pnpm test:docker:all, plus standalone Docker live lanes not covered by the aggregate when operator says "all docker tests":pnpm test:docker:live-acp-bind,pnpm test:docker:live-cli-backend, andpnpm test:docker:live-codex-harness - all Parallels install/update tests:
pnpm test:parallels:npm-update -- --jsonplus any needed individual rerun lanes fromopenclaw-parallels-smoke - all QA release validation: dispatch GitHub Actions >
QA-Lab - All Lanesagainst the release tag and require success. This is the release gate for live credentialed Matrix/Telegram channel coverage. Use a SHA only when it satisfies the workflow trust gate. Run local OpenAI/Anthropic suites or repo-backed character evals only when the operator asks for extra model coverage or a failure needs local debugging.
- Post-published beta verification roster:
node --import tsx scripts/openclaw-npm-postpublish-verify.ts <beta-version>- install/update smoke against the published beta channel
- Docker install/update coverage that exercises the published beta package
- published npm Telegram proof: dispatch Actions >
NPM Telegram Beta E2Efrommainwithpackage_spec=openclaw@<beta-version>andprovider_mode=mock-openai, approvenpm-release, and require success. This is the default button path for installed-package onboarding, Telegram setup, and real Telegram E2E against the published npm package. Use the localpnpm test:docker:npm-telegram-livelane with the matchingOPENCLAW_NPM_TELEGRAM_PACKAGE_SPECand Convex CI env only as a fallback or debugging path. - Parallels published beta install/update coverage with both OpenAI and Anthropic provider keys available
- Parallels install/update proof must keep plugin installs enabled unless the operator explicitly scopes a harness-only isolation check; a lane that disables bundled plugin installs is not valid plugin/dependency release evidence.
- targeted QA reruns only for areas touched by fixes after the full pre-npm
roster, unless the operator requests the full QA roster again. If the fix
touches live channel QA, credential plumbing, Matrix, Telegram, or the QA
harness, rerun Actions >
QA-Lab - All Lanes.
- Check all release-related build surfaces touched by the release, not only the npm package.
- For beta-style full e2e batteries, hard-cap top-level long lanes instead of letting them run indefinitely. Use host
timeout --foreground/gtimeout --foregroundcaps such as:45mforOPENCLAW_INSTALL_SMOKE_SKIP_NONROOT=1 pnpm test:install:smoke90mforpnpm test:docker:all60meach for standalone Docker live lanes180mfor local full QA live OpenAI + Anthropic rosters when explicitly requested; the default release channel QA gate is Actions >QA-Lab - All Lanes- Parallels caps from the
openclaw-parallels-smokeskill If a lane hits its cap, stop and inspect/fix the affected lane before continuing; do not continue to wait on the same process.
- Actual npm install/update phases are capped at 5 minutes. If
npm install -g, installer package install, oropenclaw updatetakes longer than 300s in release e2e, stop treating the run as healthy progress and debug the installer/updater or harness. - Serialize host build/package mutations ahead of VM lanes. Finish
pnpm build,pnpm ui:build,pnpm release:check, install smoke, and any Docker/package-prep lanes before starting Parallelsnpm packlanes; otherwisedistcan disappear during VM pack prep and produce false failures. - Include mac release readiness in preflight by running the public validation
workflow in
openclaw/openclawand the real mac preflight inopenclaw/releases-privatefor every release. - Treat the
appcast.xmlupdate onmainas part of mac release readiness, not an optional follow-up. - The workflows remain tag-based. The agent is responsible for making sure preflight runs complete successfully before any publish run starts.
- Any fix after preflight means a new commit. Delete and recreate the tag and matching GitHub release from the fixed commit, then rerun preflight from scratch before publishing. Exception: never delete or recreate a beta tag that has already been pushed or published; increment to the next beta number instead.
- For stable mac releases, generate the signed
appcast.xmlbefore uploading public release assets so the updater feed cannot lag the published binaries. - Serialize stable appcast-producing runs across tags so two releases do not
generate replacement
appcast.xmlfiles from the same stale seed. - For stable releases, rely primarily on the latest beta's broader release
workflow confidence. When promoting the matching non-beta build to npm
latest, prefer a light time-bounded verification pass: published npm postpublish verify, Docker install/update smoke, macOS-only Parallels install/update smoke, and required QA signal. Do not rerun the full Docker/Parallels matrix unless the beta evidence is stale, the stable build differs materially from beta, or the operator explicitly asks for full retesting. - If any required build, packaging step, or release workflow is red, do not say the release is ready.
Use the right auth flow
- OpenClaw publish uses GitHub trusted publishing.
- Stable npm promotion from
betatolatestuses the privateopenclaw/releases-private/.github/workflows/openclaw-npm-dist-tags.ymlworkflow becausenpm dist-tagmanagement needsNPM_TOKEN, while the public npm release workflow stays OIDC-only. - Prefer fixing the private workflow token path over any local 1Password
fallback. The desired setup is a granular npm token stored as the private
repo's
NPM_TOKENsecret, scoped to theopenclawpackage with read/write and 2FA bypass for automation. - If the private dist-tag workflow cannot promote because
NPM_TOKENis absent or stale, use the local tmux + 1Password fallback:- Start or reuse a tmux session so interactive
npm loginand OTP prompts are observable and recoverable. - Hard rule: never run
opdirectly in the main agent shell during release work. Any 1Password CLI use must happen inside that tmux session so prompts and alerts are contained and observable. - Use the 1Password item
op://Private/Npmjsfor npm credentials and OTP. Do not print passwords, tokens, or OTPs to the transcript; send them through tmux buffers, env vars scoped to the tmux command, orexpectwithlog_user 0. - Re-authenticate npm inside that tmux session with
npm login --auth-type=legacy, then confirmnpm whoamireportssteipete. - Promote with a fresh OTP:
npm dist-tag add openclaw@YYYY.M.D latest --otp "$OTP". - Verify with a cache-bypassed registry read, for example:
npm view openclaw dist-tags --json --prefer-online --cache /tmp/openclaw-npm-cache-verify-$$andnpm view openclaw@latest version dist.tarball --json --prefer-online.
- Start or reuse a tmux session so interactive
- Direct stable publishes can also use that private dist-tag workflow to point
betaat the already-publishedlatestversion when the operator wants both tags aligned immediately. - The publish run must be started manually with
workflow_dispatch. - The npm workflow and the private mac publish workflow accept
preflight_only=trueto run validation/build/package steps without uploading public release assets. - Real npm publish requires a prior successful npm preflight run id so the publish job promotes the prepared tarball instead of rebuilding it.
- Real private mac publish requires a prior successful private mac preflight run id so the publish job promotes the prepared artifacts instead of rebuilding or renotarizing them again.
- The private mac workflow also accepts
smoke_test_only=truefor branch-safe workflow smoke tests that use ad-hoc signing, skip notarization, skip shared appcast generation, and do not prove release readiness. preflight_only=trueon the npm workflow is also the right way to validate an existing tag after publish; it should keep running the build checks even when the npm version is already published.- npm validation-only preflight may still be dispatched from ordinary branches
when testing workflow changes before merge. Release checks and real publish
use only
mainorrelease/YYYY.M.D. .github/workflows/macos-release.ymlinopenclaw/openclawis now a public validation-only handoff. It validates the tag/release state and points operators to the private repo. It still rebuilds the JS outputs needed for release validation, but it does not sign, notarize, or publish macOS artifacts.openclaw/releases-private/.github/workflows/openclaw-macos-validate.ymlis the required private mac validation lane forswift test; keep it green before any real stable mac publish run starts.- Real mac preflight and real mac publish both use
openclaw/releases-private/.github/workflows/openclaw-macos-publish.yml. - The private mac validation lane runs on GitHub's standard macOS runner.
- The private mac preflight path runs on GitHub's xlarge macOS runner and uses a SwiftPM cache because the build/sign/notarize/package path is CPU-heavy.
- Private mac preflight uploads notarized build artifacts as workflow artifacts instead of uploading public GitHub release assets.
- Private smoke-test runs upload ad-hoc, non-notarized build artifacts as
workflow artifacts and intentionally skip stable
appcast.xmlgeneration. - For stable releases, npm preflight, public mac validation, private mac validation, and private mac preflight must all pass before any real publish run starts. For beta releases, npm preflight plus the selected Docker, install/update, Parallels, and release-check lanes are sufficient unless mac beta validation was explicitly requested.
- Real publish runs may be dispatched from
mainor from arelease/YYYY.M.Dbranch. For release-branch runs, the tag must be contained in that release branch, and the real publish must reuse a successful preflight from the same branch. - The release workflows stay tag-based; rely on the documented release sequence rather than workflow-level SHA pinning.
- The
npm-releaseenvironment must be approved by@openclaw/openclaw-release-managersbefore publish continues. - Mac publish uses
openclaw/releases-private/.github/workflows/openclaw-macos-publish.ymlfor private mac preflight artifact preparation and real publish artifact promotion. - Real private mac publish uploads the packaged
.zip,.dmg, and.dSYM.zipassets to the existing GitHub release inopenclaw/openclawautomatically whenOPENCLAW_PUBLIC_REPO_RELEASE_TOKENis present in the private repomac-releaseenvironment. - For stable releases, the agent must also download the signed
macos-appcast-<tag>artifact from the successful private mac workflow and then updateappcast.xmlonmain. - For beta mac releases, do not update the shared production
appcast.xmlunless a separate beta Sparkle feed exists. - The private repo targets a dedicated
mac-releaseenvironment. If the GitHub plan does not yet support required reviewers there, do not assume the environment alone is the approval boundary; rely on private repo access and CODEOWNERS until those settings can be enabled. - Do not use
NPM_TOKENor the plugin OTP flow for the OpenClaw package publish path; package publishing uses trusted publishing. - Use
NPM_TOKENonly for explicit npm dist-tag management modes, because npm does not support trusted publishing fornpm dist-tag add. @openclaw/*plugin publishes use a separate maintainer-only flow.- Only publish plugins that already exist on npm; bundled disk-tree-only plugins stay unpublished.
Fallback local mac publish
- Keep the original local macOS publish workflow available as a fallback in case CI/CD mac publishing is unavailable or broken.
- Preserve the existing maintainer workflow Peter uses: run it on a real Mac with local signing, notary, and Sparkle credentials already configured.
- Follow the private maintainer macOS runbook for the local steps:
scripts/package-mac-dist.shto build, sign, notarize, and package the app; manual GitHub release asset upload; thenscripts/make_appcast.shplus theappcast.xmlcommit tomain. scripts/package-mac-dist.shnow fails closed for release builds if the bundled app comes out with a debug bundle id, an empty Sparkle feed URL, or aCFBundleVersionbelow the canonical Sparkle build floor for that short version. For correction tags, set a higher explicitAPP_BUILD.scripts/make_appcast.shfirst usesgenerate_appcastfromPATH, then falls back to the SwiftPM Sparkle tool output underapps/macos/.build.- For stable tags, the local fallback may update the shared production
appcast.xml. - For beta tags, the local fallback still publishes the mac assets but must not
update the shared production
appcast.xmlunless a separate beta feed exists. - Treat the local workflow as fallback only. Prefer the CI/CD publish workflow when it is working.
- After any stable mac publish, verify all of the following before you call the
release finished:
- the GitHub release has
.zip,.dmg, and.dSYM.zipassets appcast.xmlonmainpoints at the new stable zip- the packaged app reports the expected short version and a numeric
CFBundleVersionat or above the canonical Sparkle build floor
- the GitHub release has
Run the release sequence
- Confirm the operator explicitly wants to cut a release.
- Choose the exact target version and git tag.
- Commit any dirty files in coherent groups, push, pull/rebase, and verify the worktree is clean.
- Pull latest
mainand confirm currentmainCI is green. - Run
/changelogfor the stable base target version onmain, commit the changelog rewrite immediately, push, and pull/rebase. For beta releases, keep the changelog heading as## YYYY.M.D, not## YYYY.M.D-beta.N. - Create
release/YYYY.M.Dfrom that post-changelogmaincommit. - Make every repo version location match the beta tag before creating it.
- Commit release preparation changes on the release branch and push the branch.
- Run the local build, Docker, and Parallels parts of the full pre-npm beta test roster from the release branch before any npm preflight or publish.
- For beta releases, skip mac app build/sign/notarize unless beta scope or a release blocker specifically requires it. For stable releases, include the mac app, signing, notarization, and appcast path.
- Confirm the target npm version is not already published.
- Create and push the git tag from the release branch.
- Create or refresh the matching GitHub release.
- Dispatch Actions >
QA-Lab - All Lanesagainst the release tag and wait for the mock parity, live Matrix, and live Telegram credentialed-channel lanes to pass. - Start
.github/workflows/openclaw-npm-release.ymlfrom the release branch withpreflight_only=trueand choose the intendednpm_dist_tag(betadefault;latestonly for an intentional direct stable publish). Wait for it to pass. Save that run id because the real publish requires it to reuse the prepared npm tarball. - For stable releases, start
.github/workflows/macos-release.ymlinopenclaw/openclawand wait for the public validation-only run to pass. - For stable releases, start
openclaw/releases-private/.github/workflows/openclaw-macos-validate.ymlwith the same tag and wait for the private mac validation lane to pass. - For stable releases, start
openclaw/releases-private/.github/workflows/openclaw-macos-publish.ymlwithpreflight_only=trueand wait for it to pass. Save that run id because the real publish requires it to reuse the notarized mac artifacts. - If any preflight or validation run fails, fix the issue on a new commit, delete the tag and matching GitHub release, recreate them from the fixed commit, and rerun all relevant preflights from scratch before continuing. Never reuse old preflight results after the commit changes. For pushed or published beta tags, do not delete/recreate; increment to the next beta tag.
- Start
.github/workflows/openclaw-npm-release.ymlfrom the same branch with the same tag for the real publish, choosenpm_dist_tag(betadefault,latestonly when you intentionally want direct stable publish), keep it the same as the preflight run, and pass the successful npmpreflight_run_id. - Wait for
npm-releaseapproval from@openclaw/openclaw-release-managers. - Run postpublish verification:
node --import tsx scripts/openclaw-npm-postpublish-verify.ts <published-version>. - Run the post-published beta verification roster. If any lane fails after
the beta tag/package is pushed or published, fix, commit/push/pull,
increment to the next beta tag, and restart at the full pre-npm beta test
roster for the new beta. The roster includes the manual Actions >
NPM Telegram Beta E2Eworkflow against the exact published beta package. If a pre-npm lane fails before any tag/package leaves the machine, fix and rerun the same intended beta attempt. Repeat up to the operator's authorized beta-attempt limit, normally 4. - Announce the beta/stable release on Discord best-effort using Peter's bot
token from
.profile. - If the operator requested beta only, stop after beta verification and the announcement.
- If the stable release was published to
beta, use the light stable promotion roster when the matching beta already carried the full confidence pass: published npm postpublish verify, Docker install/update smoke, macOS-only Parallels install/update smoke, and required QA signal. Then start the privateopenclaw/releases-private/.github/workflows/openclaw-npm-dist-tags.ymlworkflow to promote that stable version frombetatolatest, then verifylatestnow points at that version. - If the stable release was published directly to
latestandbetashould follow it, start that same private dist-tag workflow to pointbetaat the stable version, then verify bothlatestandbetapoint at that version. - For stable releases, start
openclaw/releases-private/.github/workflows/openclaw-macos-publish.ymlfor the real publish with the successful private macpreflight_run_idand wait for success. - Verify the successful real private mac run uploaded the
.zip,.dmg, and.dSYM.zipartifacts to the existing GitHub release inopenclaw/openclaw. - For stable releases, download
macos-appcast-<tag>from the successful private mac run, updateappcast.xmlonmain, and verify the feed. Merge or cherry-pick release branch changes back tomainafter stable succeeds. - For beta releases, publish the mac assets only when intentionally requested;
expect no shared production
appcast.xmlartifact and do not update the shared production feed unless a separate beta feed exists. - After publish, verify npm and the attached release artifacts.
GHSA advisory work
- Use
openclaw-ghsa-maintainerfor GHSA advisory inspection, patch/publish flow, private-fork validation, and GHSA API-specific publish checks.