.agents/skills/publish/SKILL.md
You are the release manager for oh-my-opencode. Execute the FULL publish workflow from start to finish.
Publishing is not complete until the Discord release announcement has been attempted.
You MUST receive a version bump type from the user. Valid options:
patch: Bug fixes, backward-compatible (1.1.7 ā 1.1.8)minor: New features, backward-compatible (1.1.7 ā 1.2.0)major: Breaking changes (1.1.7 ā 2.0.0)If the user did not provide a bump type argument, STOP IMMEDIATELY and ask:
"To proceed with deployment, please specify a version bump type:
patch,minor, ormajor"
DO NOT PROCEED without explicit user confirmation of bump type.
Before doing ANYTHING else, create a detailed todo list using TodoWrite:
[
{ "id": "confirm-bump", "content": "Confirm version bump type with user (patch/minor/major)", "status": "in_progress", "priority": "high" },
{ "id": "check-uncommitted", "content": "Check for uncommitted changes and commit if needed", "status": "pending", "priority": "high" },
{ "id": "sync-remote", "content": "Sync with remote (pull --rebase && push if unpushed commits)", "status": "pending", "priority": "high" },
{ "id": "run-workflow", "content": "Trigger GitHub Actions publish workflow", "status": "pending", "priority": "high" },
{ "id": "wait-workflow", "content": "Wait for workflow completion (poll every 30s)", "status": "pending", "priority": "high" },
{ "id": "verify-and-preview", "content": "Verify release created + preview auto-generated changelog & contributor thanks", "status": "pending", "priority": "high" },
{ "id": "draft-summary", "content": "Draft enhanced release summary (mandatory for all release types)", "status": "pending", "priority": "high" },
{ "id": "apply-summary", "content": "Prepend enhanced summary to release", "status": "pending", "priority": "high" },
{ "id": "discord-announce", "content": "MANDATORY: post release announcement to Discord channel immediately after release notes are finalized", "status": "pending", "priority": "high" },
{ "id": "verify-npm", "content": "Verify npm package published successfully", "status": "pending", "priority": "high" },
{ "id": "wait-platform-workflow", "content": "Wait for publish-platform workflow completion", "status": "pending", "priority": "high" },
{ "id": "verify-platform-binaries", "content": "Verify all 7 platform binary packages published", "status": "pending", "priority": "high" },
{ "id": "final-confirmation", "content": "Final confirmation to user with links", "status": "pending", "priority": "low" }
]
Mark each todo as in_progress when starting, completed when done. ONE AT A TIME.
If bump type provided as argument, confirm with user:
"Version bump type:
{bump}. Proceed? (y/n)"
Wait for user confirmation before proceeding.
Run: git status --porcelain
Check if there are unpushed commits:
git log origin/master..HEAD --oneline
If there are unpushed commits, you MUST sync before triggering workflow:
git pull --rebase && git push
This ensures the GitHub Actions workflow runs on the latest code including all local commits.
Run the publish workflow:
gh workflow run publish -f bump={bump_type}
Wait 3 seconds, then get the run ID:
gh run list --workflow=publish --limit=1 --json databaseId,status --jq '.[0]'
Poll workflow status every 30 seconds until completion:
gh run view {run_id} --json status,conclusion --jq '{status: .status, conclusion: .conclusion}'
Status flow: queued ā in_progress ā completed
IMPORTANT: Use polling loop, NOT sleep commands.
If conclusion is failure, show error and stop:
gh run view {run_id} --log-failed
Two goals: confirm the release exists, then show the user what the workflow already generated.
# Pull latest (workflow committed version bump)
git pull --rebase
NEW_VERSION=$(node -p "require('./package.json').version")
# Verify release exists on GitHub
gh release view "v${NEW_VERSION}" --json tagName,url --jq '{tag: .tagName, url: .url}'
After verifying, generate a local preview of the auto-generated content:
bun run script/generate-changelog.ts
The following content is ALREADY included in the release automatically:
- Commit changelog (grouped by feat/fix/refactor)
- Contributor thank-you messages (for non-team contributors)
You do NOT need to write any of this. It's handled.
For all release types, an enhanced summary is required ā I'll draft one in the next step.
Wait for the user to acknowledge before proceeding.
If the user already confirmed the publish workflow and did not explicitly ask to review the generated changelog before release-note editing, treat the publish confirmation as sufficient acknowledgement and continue. Do not end the assistant turn here. </agent-instruction>
| Release Type | Action |
|---|---|
| patch | MANDATORY. Draft a concise bug-fix / change summary. Do NOT proceed without one. |
| minor | MANDATORY. Draft a concise feature summary. Do NOT proceed without one. |
| major | MANDATORY. Draft a full release narrative with migration notes if applicable. Do NOT proceed without one. |
You are writing the headline layer ā a product announcement that sits ABOVE the auto-generated commit log. Think "release blog post", not "git log".
<rules> - NEVER duplicate commit messages. The auto-generated section already lists every commit. - NEVER write generic filler like "Various bug fixes and improvements" or "Several enhancements". - ALWAYS focus on USER IMPACT: what can users DO now that they couldn't before? - ALWAYS group by THEME or CAPABILITY, not by commit type (feat/fix/refactor). - ALWAYS use concrete language: "You can now do X" not "Added X feature". </rules> <examples> <bad title="Commit regurgitation ā DO NOT do this"> ## What's New - feat(auth): add JWT refresh token rotation - fix(auth): handle expired token edge case - refactor(auth): extract middleware </bad> <good title="User-impact narrative ā DO this"> ## š Smarter AuthenticationToken refresh is now automatic and seamless. Sessions no longer expire mid-task ā the system silently rotates credentials in the background. If you've been frustrated by random logouts, this release fixes that. </good>
<bad title="Vague filler ā DO NOT do this"> ## Improvements - Various performance improvements - Bug fixes and stability enhancements </bad> <good title="Specific and measurable ā DO this"> ## ā” 3x Faster Rule ParsingRules are now cached by file modification time. If your project has 50+ rule files, you'll notice startup is noticeably faster ā we measured a 3x improvement in our test suite. </good> </examples>
/tmp/release-summary-v${NEW_VERSION}.md.# Write your draft here
cat > /tmp/release-summary-v${NEW_VERSION}.md << 'SUMMARY_EOF'
{your_enhanced_summary}
SUMMARY_EOF
cat /tmp/release-summary-v${NEW_VERSION}.md
If the user already confirmed the publish workflow and did not explicitly request a release-note review hold, proceed to Step 7 after presenting the draft. Do not stop before Step 7.5, because the Discord announcement is mandatory. </agent-instruction>
This step is MANDATORY. The enhanced summary from Step 6 must always be applied.
<architecture> The final release note structure:āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
ā Enhanced Summary (from Step 6) ā ā You wrote this
ā - Theme-based, user-impact focused ā
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā¤
ā --- (separator) ā
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā¤
ā Auto-generated Commit Changelog ā ā Workflow wrote this
ā - feat/fix/refactor grouped ā
ā - Contributor thank-you messages ā
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
# 1. Fetch existing auto-generated body
EXISTING_BODY=$(gh release view "v${NEW_VERSION}" --json body --jq '.body')
# 2. Combine: enhanced summary on top, auto-generated below
{
cat /tmp/release-summary-v${NEW_VERSION}.md
echo ""
echo "---"
echo ""
echo "$EXISTING_BODY"
} > /tmp/final-release-v${NEW_VERSION}.md
# 3. Update the release (additive only)
gh release edit "v${NEW_VERSION}" --notes-file /tmp/final-release-v${NEW_VERSION}.md
# 4. Confirm
echo "ā
Release v${NEW_VERSION} updated with enhanced summary."
gh release view "v${NEW_VERSION}" --json url --jq '.url'
After the release notes are finalized, post them to the Discord channel. This step is mandatory for every publish run.
<hard-gate> The workflow is not complete until this step has either: 1. Sent a Discord message successfully and recorded the message ID, or 2. Failed after `agent-discord auth extract` plus one send retry, with the failure reported to the user.Never skip this step because the release summary was awaiting approval. If the user already confirmed the publish, continue through Discord before stopping. </hard-gate>
<agent-discord-instruction> 1. Ensure Discord auth is available: ```bash agent-discord auth extract ```agent-discord message list 1454708427392680067 --limit 5
1454708427392680067 matching the style of previous announcements. The message should follow this structure:@here
š **oh-my-opencode v{VERSION} ā {Short Tagline}**
**Feature 1** ā one-line description.
**Feature 2** ā one-line description.
**Feature 3** ā one-line description.
Plus {summary of remaining changes}.
š¦ Install / upgrade:
`bun i -g oh-my-opencode@{VERSION}` (or `npm`)
š Full release notes: {RELEASE_URL}
RELEASE_URL=$(gh release view "v${NEW_VERSION}" --json url --jq '.url')
agent-discord message send 1454708427392680067 "{your message following the style above}"
If the message fails to send, warn the user and continue ā do NOT block the publish workflow on Discord errors. </agent-discord-instruction>
Poll npm registry until the new version appears:
npm view oh-my-opencode version
Compare with expected version. If not matching after 2 minutes, warn user about npm propagation delay.
The main publish workflow triggers a separate publish-platform workflow for platform-specific binaries.
gh run list --workflow=publish-platform --limit=1 --json databaseId,status,conclusion --jq '.[0]'
gh run view {platform_run_id} --json status,conclusion --jq '{status: .status, conclusion: .conclusion}'
IMPORTANT: Use polling loop, NOT sleep commands.
If conclusion is failure, show error logs:
gh run view {platform_run_id} --log-failed
After publish-platform workflow completes, verify all 7 platform packages are published:
PLATFORMS="darwin-arm64 darwin-x64 linux-x64 linux-arm64 linux-x64-musl linux-arm64-musl windows-x64"
for PLATFORM in $PLATFORMS; do
npm view "oh-my-opencode-${PLATFORM}" version
done
All 7 packages should show the same version as the main package (${NEW_VERSION}).
Expected packages:
| Package | Description |
|---|---|
oh-my-opencode-darwin-arm64 | macOS Apple Silicon |
oh-my-opencode-darwin-x64 | macOS Intel |
oh-my-opencode-linux-x64 | Linux x64 (glibc) |
oh-my-opencode-linux-arm64 | Linux ARM64 (glibc) |
oh-my-opencode-linux-x64-musl | Linux x64 (musl/Alpine) |
oh-my-opencode-linux-arm64-musl | Linux ARM64 (musl/Alpine) |
oh-my-opencode-windows-x64 | Windows x64 |
If any platform package version doesn't match, warn the user and suggest checking the publish-platform workflow logs.
Report success to user with:
gh auth loginRespond to user in English.