release
Release Skill
Creates a release by comparing dev to main, generating changelog entries from commits, bumping the version, and creating a PR. After the tag is pushed and the release workflow finishes building binaries, updates the Homebrew formula with the real SHA256 values from the published checksums.txt, syncs the coleam00/homebrew-archon tap, and verifies the end-to-end install path via /test-release.
⚠️ CRITICAL — Homebrew formula SHAs cannot be known until after the release workflow builds binaries.
The
versionfield inhomebrew/archon.rband thesha256fields must be updated atomically. Never update one without the other.The correct sequence is:
- Tag is pushed → release workflow fires → binaries built →
checksums.txtuploaded- Fetch
checksums.txtfrom the published release- Parse the SHA256 per platform
- Update
homebrew/archon.rbwith the new version AND the new SHAs in a single commit- Sync to the
coleam00/homebrew-archon/Formula/archon.rbtap repoUpdating the formula's
versionfield without also updating thesha256values creates a stale, misleading formula that looks valid but produces checksum mismatches on install. This has happened before (v0.3.0: version updated to 0.3.0 but SHAs were still from v0.2.13). Always do both or neither.
Process
Step 1: Validate State
# Must be on dev branch with clean working tree
git checkout dev
git pull origin dev
git status --porcelain # must be empty
git fetch origin main
If not on dev or working tree is dirty, abort with a clear message.
Step 1.5: Pre-flight compiled-binary smoke test (MANDATORY before any other step)
Why this is first: releases have ended up with zero working binaries because a module-init crash or bundler bug only surfaces in
bun build --compileoutput, not inbun run. CI catches it — but only AFTER the tag is pushed and a GitHub Release is created. By then the damage (empty release, brokenreleases/latest, brokeninstall.sh) is already live. Failing here, before any user-visible change, keeps the blast radius at "no release was cut."
Run locally on the native target. This takes ~15-30s and is cheaper than discovering the problem after tag+release.
# Guard: only run this for Node/Bun projects with a CLI entry point + build-binaries script.
if [ -f scripts/build-binaries.sh ] && [ -f packages/cli/src/cli.ts ]; then
TMP_BINARY=$(mktemp)
trap "rm -f $TMP_BINARY" EXIT
# Compile for the native target only (not full cross-compile — that's CI's job).
# Match the real release flags so any bundler quirk reproduces locally.
bun build \
--compile \
--minify \
--target=bun \
--outfile="$TMP_BINARY" \
packages/cli/src/cli.ts
# Smoke test: the binary must start and exit 0 on a safe, non-interactive command.
# Use `--help` (NOT `version`). The `version` command's compiled-binary code
# path depends on BUNDLED_IS_BINARY=true, which is set by scripts/build-binaries.sh
# — but we're doing a bare `bun build --compile` here to keep the smoke fast,
# so BUNDLED_IS_BINARY is still `false`. That sends `version` down the dev
# branch of version.ts which tries to read package.json from a path that only
# exists in node_modules, producing a false-positive ENOENT. `--help` has no
# such dev/binary branch and exercises the same module-init graph we're
# actually testing. Must NOT touch network, database, or require env vars.
if ! "$TMP_BINARY" --help > /tmp/archon-preflight.log 2>&1; then
echo "ERROR: compiled binary crashed at startup"
cat /tmp/archon-preflight.log
echo ""
echo "This usually means a dependency has a module-init-time side effect that"
echo "fails in a compiled binary context (readFileSync of a path that only"
echo "exists in node_modules, etc.). Fix before cutting the release — do NOT"
echo "proceed to version bump."
exit 1
fi
# Also grep for known crash markers that exit 0 but print a fatal error
# (some module-init errors are caught by top-level try/catch but still log).
if grep -qE "Expected CommonJS module|TypeError:|ReferenceError:|SyntaxError:" /tmp/archon-preflight.log; then
echo "ERROR: compiled binary emitted a runtime error despite exit 0"
cat /tmp/archon-preflight.log
exit 1
fi
echo "Pre-flight binary smoke: PASSED"
fi
If this fails, abort the release entirely — do not bump version, do not modify CHANGELOG, do not create a PR. Surface the error to the user, point at the failing output, and stop. Recovery is: fix the bundler / dependency issue on a feature branch, merge to dev, re-run /release.
Common failure modes this catches:
- Bun
--bytecodeflag producing broken bytecode for the current module graph - A dependency (e.g. an SDK) reading
package.jsonor other files at module top level via paths that resolve fine innode_modules/but not next to a compiled binary - Circular imports that break under minification but work under plain
bun run - A newly added package that ships CJS with an unusual wrapper shape
Step 2: Detect Stack and Current Version
Detect the project's package manager and version file:
- Check for
pyproject.toml— Python project, version inversion = "x.y.z" - Check for
package.json— Node/Bun project, version in"version": "x.y.z" - Check for
Cargo.toml— Rust project, version inversion = "x.y.z" - Check for
go.mod— Go project (version from git tags only, no file to bump)
If none found, abort: "Could not detect project stack — no version file found."
Read the current version from the detected file.
Step 3: Determine Version Bump
Bump rules based on argument:
- No argument or
patch(default):0.1.0 -> 0.1.1 minor:0.1.3 -> 0.2.0major:0.3.5 -> 1.0.0
Step 4: Collect Commits
# Get all commits on dev that aren't on main
git log main..dev --oneline --no-merges
If no new commits, abort: "Nothing to release — dev is up to date with main."
Step 5: Draft Changelog Entries
Read the commit messages and the actual diffs (git diff main..dev) to understand what changed.
Categorize into Keep a Changelog sections:
- Added — new features, new files, new capabilities
- Changed — modifications to existing behavior
- Fixed — bug fixes
- Removed — deleted features or code
Writing rules:
- Write entries as a human would — clear, concise, user-facing language
- Do NOT just copy commit messages verbatim — rewrite them into proper changelog entries
- Group related commits into single entries where it makes sense
- Include PR numbers in parentheses when available:
(#12) - Each entry should start with a noun or gerund describing WHAT changed
- Skip internal-only changes (CI tweaks, typo fixes) unless they affect behavior
- One blank line between sections
Step 6: Update Files
-
Version file — update version to new value:
package.json: update"version": "x.y.z"pyproject.toml: updateversion = "x.y.z"Cargo.toml: updateversion = "x.y.z"
-
Workspace version sync (monorepo only):
- If
scripts/sync-versions.shexists, runbash scripts/sync-versions.shto sync allpackages/*/package.jsonversions to match the root version.
- If
-
Lockfile refresh (stack-dependent):
package.json+bun.lock: runbun installpackage.json+package-lock.json: runnpm install --package-lock-onlypyproject.toml+uv.lock: runuv lock --quietCargo.toml: runcargo update --workspace
-
CHANGELOG.md— prepend new version section:
## [x.y.z] - YYYY-MM-DD
One-line summary of the release.
### Added
- Entry one (#PR)
- Entry two (#PR)
### Changed
- Entry one (#PR)
### Fixed
- Entry one (#PR)
Move any content under [Unreleased] into the new version section. Leave [Unreleased] header with nothing under it.
Step 7: Present for Review
Show the user:
- The detected stack and version file
- The version bump (old -> new)
- The full changelog section that will be added
- The list of commits being included
Ask: "Does this look good? I'll commit and create the PR."
Step 8: Commit and PR
Only after user approval:
# Stage version file, workspace packages, lockfile, and changelog
git add <version-file> packages/*/package.json <lockfile> CHANGELOG.md
git commit -m "Release x.y.z"
# Push dev
git push origin dev
# Create PR: dev -> main
gh pr create --base main --head dev \
--title "Release x.y.z" \
--body "$(cat <<'EOF'
## Release x.y.z
{changelog section content}
---
Merging this PR releases x.y.z to main.
EOF
)"
Return the PR URL to the user.
Step 9: Tag, Release, and Sync After Merge
After the PR is merged (either by the user or via gh pr merge):
# Fetch the merge commit on main
git fetch origin main
# Tag the merge commit
git tag vx.y.z origin/main
git push origin vx.y.z
# Create a GitHub Release from the tag (uses changelog content as release notes)
gh release create vx.y.z --title "vx.y.z" --notes "{changelog section content without the ## header}"
# Sync dev with main so both branches are identical
git checkout dev
git pull origin main
git push origin dev
Important: This sync ensures dev has the merge commit from main. Without it, dev and main diverge. The CI
update-homebrewjob only pushes the formula commit to dev — it does not bring the PR merge commit onto dev. This manualgit pull origin mainis what ensures dev has the merge commit.
The GitHub Release is distinct from the git tag — without it, the release won't appear on the repository's Releases page. Always create it.
If the user merges the PR themselves and comes back, still offer to tag, release, and sync.
Step 10: Wait for Release Workflow and Update Homebrew Formula
Note: The
update-homebrewCI job in.github/workflows/release.ymlruns automatically after the release job and handles the formula update + push to dev (part of Step 10). Step 11 (tap sync tocoleam00/homebrew-archon) is always manual. Check the Actions tab before running Step 10 manually.
After the tag is pushed, .github/workflows/release.yml builds platform binaries and uploads them to the GitHub release. This takes 5-10 minutes. The Homebrew formula SHA256 values cannot be known until these binaries exist.
Wait for all assets to appear on the release:
echo "Waiting for release workflow to finish uploading binaries..."
WORKFLOW_FAILED=0
for i in {1..30}; do
ASSET_COUNT=$(gh release view "vx.y.z" --repo coleam00/Archon --json assets --jq '.assets | length')
# Expect 7 assets: 5 binaries (darwin-arm64, darwin-x64, linux-arm64, linux-x64, windows-x64.exe) + archon-web.tar.gz + checksums.txt
if [ "$ASSET_COUNT" -ge 7 ]; then
echo "All $ASSET_COUNT assets uploaded"
break
fi
# Short-circuit: if the release workflow itself has failed, stop waiting.
# Hanging for 15 min when CI already crashed just delays the recovery path.
WORKFLOW_STATUS=$(gh run list --workflow release.yml --event push --limit 1 --json conclusion,status --jq '.[0] | "\(.status)|\(.conclusion)"')
if [[ "$WORKFLOW_STATUS" == "completed|failure" ]]; then
echo "Release workflow FAILED — no point waiting longer"
WORKFLOW_FAILED=1
break
fi
echo " Assets so far: $ASSET_COUNT/7 — waiting 30s (attempt $i/30)..."
sleep 30
done
if [ "$WORKFLOW_FAILED" -eq 1 ] || [ "$ASSET_COUNT" -lt 7 ]; then
# Triage: rerun once in case it's transient, then check again.
RUN_ID=$(gh run list --workflow release.yml --event push --limit 1 --json databaseId --jq '.[0].databaseId')
echo "Release workflow failed on run $RUN_ID. Rerunning failed jobs once to confirm..."
gh run rerun "$RUN_ID" --failed
gh run watch "$RUN_ID" --exit-status --interval 30 || true
# Re-check asset count + run status after rerun.
ASSET_COUNT=$(gh release view "vx.y.z" --repo coleam00/Archon --json assets --jq '.assets | length')
if [ "$ASSET_COUNT" -ge 7 ]; then
echo "Rerun succeeded — all assets now present"
else
echo ""
echo "===== DETERMINISTIC CI FAILURE ====="
echo "The release workflow failed on two consecutive runs. This is NOT a flake."
echo "The tag and release exist but have no (or incomplete) assets."
echo ""
echo "install.sh and similar 'releases/latest' paths are now 404-ing."
echo "Proceeding with Homebrew/tap sync would publish a formula pointing at"
echo "missing or inconsistent binaries."
echo ""
echo "Jump to the 'Recovery: deterministic release CI failure' section at the"
echo "bottom of this skill and execute it. Do NOT continue past this point."
exit 1
fi
fi
Fetch checksums.txt and extract SHA256 values:
TMP_DIR=$(mktemp -d)
gh release download "vx.y.z" --repo coleam00/Archon --pattern "checksums.txt" --dir "$TMP_DIR"
DARWIN_ARM64_SHA=$(awk '/archon-darwin-arm64$/ {print $1}' "$TMP_DIR/checksums.txt")
DARWIN_X64_SHA=$(awk '/archon-darwin-x64$/ {print $1}' "$TMP_DIR/checksums.txt")
LINUX_ARM64_SHA=$(awk '/archon-linux-arm64$/ {print $1}' "$TMP_DIR/checksums.txt")
LINUX_X64_SHA=$(awk '/archon-linux-x64$/ {print $1}' "$TMP_DIR/checksums.txt")
# Sanity check — all four must be present and non-empty
for var in DARWIN_ARM64_SHA DARWIN_X64_SHA LINUX_ARM64_SHA LINUX_X64_SHA; do
if [ -z "${!var}" ]; then
echo "ERROR: $var is empty — checksums.txt may be malformed"
cat "$TMP_DIR/checksums.txt"
exit 1
fi
done
rm -rf "$TMP_DIR"
Update homebrew/archon.rb in the main repo atomically with version AND SHAs:
Rewrite the formula file using the exact template below. Do NOT edit in place with sed — the whole file should be regenerated from this template so there is zero risk of partial updates.
cat > homebrew/archon.rb << EOF
# Homebrew formula for Archon CLI
# To install: brew install coleam00/archon/archon
#
# This formula downloads pre-built binaries from GitHub releases.
# For development, see: https://github.com/coleam00/Archon
class Archon < Formula
desc "Remote agentic coding platform - control AI assistants from anywhere"
homepage "https://github.com/coleam00/Archon"
version "x.y.z"
license "MIT"
on_macos do
on_arm do
url "https://github.com/coleam00/Archon/releases/download/v#{version}/archon-darwin-arm64"
sha256 "${DARWIN_ARM64_SHA}"
end
on_intel do
url "https://github.com/coleam00/Archon/releases/download/v#{version}/archon-darwin-x64"
sha256 "${DARWIN_X64_SHA}"
end
end
on_linux do
on_arm do
url "https://github.com/coleam00/Archon/releases/download/v#{version}/archon-linux-arm64"
sha256 "${LINUX_ARM64_SHA}"
end
on_intel do
url "https://github.com/coleam00/Archon/releases/download/v#{version}/archon-linux-x64"
sha256 "${LINUX_X64_SHA}"
end
end
def install
binary_name = case
when OS.mac? && Hardware::CPU.arm?
"archon-darwin-arm64"
when OS.mac? && Hardware::CPU.intel?
"archon-darwin-x64"
when OS.linux? && Hardware::CPU.arm?
"archon-linux-arm64"
when OS.linux? && Hardware::CPU.intel?
"archon-linux-x64"
end
bin.install binary_name => "archon"
end
test do
# Basic version check - archon version should exit with 0 on success
assert_match version.to_s, shell_output("#{bin}/archon version")
end
end
EOF
Commit the formula update to main, then sync back to dev:
git checkout main
git pull origin main
git add homebrew/archon.rb
git commit -m "chore(homebrew): update formula to vx.y.z"
git push origin main
# Sync dev with main so the formula update is on both branches
git checkout dev
git pull origin main
git push origin dev
Step 11: Sync the Homebrew Tap Repo
The coleam00/homebrew-archon repository hosts the actual tap formula that Homebrew reads when users run brew tap coleam00/archon && brew install coleam00/archon/archon. The file coleam00/Archon/homebrew/archon.rb is the source-of-truth template; the file coleam00/homebrew-archon/Formula/archon.rb is what users actually install from. These must be kept in sync.
TAP_DIR=$(mktemp -d)
git clone git@github.com:coleam00/homebrew-archon.git "$TAP_DIR"
cp homebrew/archon.rb "$TAP_DIR/Formula/archon.rb"
cd "$TAP_DIR"
if git diff --quiet; then
echo "Tap formula already matches — no sync needed"
else
git add Formula/archon.rb
git commit -m "chore: sync formula to vx.y.z"
git push origin main
fi
cd -
rm -rf "$TAP_DIR"
If the git clone fails with a permissions error, the user running the release skill does not have push access to coleam00/homebrew-archon. Ask them to request push access from the repo owner, or to perform the sync manually via the GitHub web UI. Do not skip this step silently — the release is not complete until the tap is synced.
Step 12: Verify the Release End-to-End
After the formula is synced, the final verification step is to actually install the released binary via Homebrew and run smoke tests. Use the test-release skill:
/test-release brew x.y.z
This will:
- Install via
brew tap coleam00/archon && brew install coleam00/archon/archon - Verify the binary reports the correct version and
Build: binary - Verify bundled workflows load
- Verify the SDK spawn path works (a minimal assist workflow)
- Verify the env-leak gate is active (if shipped in this release)
- Uninstall cleanly
- Produce a PASS/FAIL report
If /test-release brew fails, the release is not ready to announce. File a hotfix issue for whatever broke, cut x.y.z+1 with the fix, and re-run this skill. Do NOT advertise a release that fails test-release.
Also run /test-release curl-mac x.y.z to cover the curl install path. The two install paths test slightly different things (Homebrew tests the tap formula, curl tests install.sh and checksums from the release) and both need to work for users to have a reliable install experience.
If you have a VPS available, also run /test-release curl-vps x.y.z <vps-target> to verify the Linux binary.
Recovery: deterministic release CI failure
Reached here because Step 10 detected two consecutive workflow failures. The tag vx.y.z is pushed, the GitHub release exists, but assets are missing or incomplete. Every install.sh run currently resolves releases/latest to this broken release and 404s on download. Homebrew users are safe because Step 10's atomic formula update was blocked.
Do not re-run the release workflow a third time hoping it succeeds. If the failure was reproducible twice, it's a code bug — you need to ship code to fix it.
Immediate mitigation (restore install.sh)
Delete the GitHub Release so releases/latest falls back to the previous version. Keep the git tag — tag immutability matters and there are no shipped artifacts pointing at it anyway.
gh release delete "vx.y.z" --yes
# Do NOT delete the tag:
# git push --delete origin vx.y.z ← do not run
# Tag stays so git history records the attempt; no release means no assets
# means releases/latest resolves to the prior working release.
Verify:
gh api repos/coleam00/Archon/releases/latest --jq '.tag_name'
# should now print the prior version (e.g. v0.3.6), not vx.y.z
Diagnose
The release workflow logs tell you which target failed and at what stage (compile vs. smoke-test vs. upload):
gh run list --workflow release.yml --limit 2 --json databaseId,conclusion
gh run view <RUN_ID> --log-failed
Common causes:
- Bundler/bytecode bug — Bun
--bytecodeproduces invalid output for the current module graph. Symptom:TypeError: Expected CommonJS module to have a function wrapperat binary startup. Historically caused by a new dependency's CJS/ESM shape interacting with--bytecode— dropping the flag or lazy-importing the offending module has been the fix. - Module-init crash — a dependency does
readFileSync('package.json')or similar at module top level via a path that exists innode_modules/but not next to a compiled binary. Symptom: every binary subcommand crashes immediately; error typically mentions a missing file adjacent toprocess.execPath. Fix by lazy-importing the dependency behind the code path that actually uses it. - Smoke-test timeout on Windows — not actually a bug in the code; the Windows runner is slow. Rerun once; if it recurs, bump the test timeout.
Step 1.5 now runs a local compiled-binary smoke test before any user-visible step. If the failure mode above reproduces locally, you've found it. If it doesn't, the bug is platform-specific (Windows cross-compile, Linux glibc, etc.) and you need the CI logs.
Fix and re-release as the NEXT patch
Do not reuse vx.y.z. Cut vx.y.(z+1) (or next-minor if warranted) with the fix. Rationale:
- Tag immutability:
vx.y.zis already recorded in git history and release cache - Semver clarity: users and tooling should see a new version number when the bits change
- Audit trail: "v0.3.7 was cut but had no shipped binaries; v0.3.8 is the first release with " is cleaner than rewriting v0.3.7
Steps:
- Cut a fix branch off dev, implement the fix, PR to dev, merge.
- Re-run
/release(it will bump to the next patch — e.g.0.3.8— automatically). - Step 1.5's pre-flight smoke will catch the same bug locally if the fix didn't actually fix it. Iterate until it passes before tagging.
CHANGELOG note for the hotfix release
Include a line in the new release's CHANGELOG that references the broken prior version so users understand why there's no binary artifact under that tag:
### Fixed
- **First release with working compiled binaries after vx.y.z's <bug>.** vx.y.z was tagged but its binary smoke test failed deterministically (see RUN_ID in CI history). The tag is preserved for history; this release (vx.y.(z+1)) is the first with shipped binaries. `install.sh` and Homebrew were never updated to vx.y.z, so users were not exposed to the broken state.
What NOT to do
- Do not force-push or rewrite the tag. Once a tag exists, it's a public promise of that SHA. Deleting and re-creating to a different SHA is tag-spoofing and breaks any downstream that cached the original.
- Do not skip this recovery path to "just push more binaries to the broken release". The release exists with a specific commit SHA; uploading binaries built from a newer SHA creates binary/source drift that is hard to diagnose later.
- Do not update the Homebrew formula before v0.3.(z+1) is fully shipped. The formula should always point at a version with all 7 assets uploaded and
/test-release brewpassing.
Important Rules
- NEVER force push
- NEVER skip Step 1.5 (pre-flight compiled-binary smoke). If the stack is a Bun/Node project with a build-binaries script, the
bun build --compilesmoke test runs before version bump, PR, or tag. Skipping it means every bundler regression or module-init crash only surfaces after the tag is pushed — by which pointreleases/latestis already 404-ing for every user. The ~30s cost is paid to keep the failure mode local. - If Step 1.5 fails, abort the release and fix the underlying issue on a feature branch. Do not "just skip it" and hope CI doesn't repro the problem.
- NEVER skip the review step — always show the changelog before committing
- NEVER include "Co-Authored-By: Claude" or any AI attribution in the commit
- NEVER add emoji to changelog entries unless the user asks
- If the user says "ship it" without specifying bump type, default to patch
- The commit message is just
Release x.y.z— clean and simple - NEVER update
homebrew/archon.rbversion field without also updating thesha256values. They must move together atomically. The correct SHAs only exist after the release workflow finishes building binaries — see Step 10. Updating the version field alone produces a stale formula that looks valid but causes checksum mismatches on install. - NEVER skip Step 11 (tap sync). The
coleam00/Archon/homebrew/archon.rbfile is only a template; users install fromcoleam00/homebrew-archon/Formula/archon.rb. If you update one without the other, users get stale or wrong data. - NEVER announce a release that failed
/test-release brew. A release that installs but crashes on first invocation is worse than no release — it burns user trust. If the release verification fails, cut a hotfix before telling anyone the release exists.