.agents/skills/create-draft-release-notes/SKILL.md
Create a GitHub draft release, organize the generated notes by conventional commit type, and save the organized body back to the draft. Preserve each release note item exactly; only split accidentally joined bullets, move bullets into sections, and adjust headings.
Input: a release tag/title such as v2.0.6. If title and tag differ, ask for the tag.
Resolve repo as <owner>/<repo>.
Prefer an explicit repo from the user. Otherwise infer the current project's main GitHub repository from project metadata or the current GitHub remote. For npm projects, package.json repository is a useful signal; in monorepos, inspect the package or project being released rather than assuming the workspace root. Ignore subdirectory metadata such as repository.directory because GitHub releases are repository-level. If the repo is ambiguous, ask.
Set variables:
repo="<owner>/<repo>"
release_tag="v2.0.6"
release_title="$release_tag"
Verify access and ensure the release does not already exist:
gh auth status
gh repo view "$repo" --json nameWithOwner --jq '.nameWithOwner'
gh release view "$release_tag" -R "$repo" --json tagName,isDraft,url
If the release exists, stop unless the user explicitly asked to update that draft.
Infer the previous tag:
previous_tag="$(gh release list -R "$repo" --exclude-drafts --exclude-pre-releases --limit 1 --json tagName --jq '.[0].tagName')"
gh release list -R "$repo" --exclude-drafts --exclude-pre-releases --limit 5
Ask for confirmation if the previous tag is missing, surprising, or part of a non-standard range.
Before creating anything, state the repo and range: previous_tag -> release_tag. If the user did not explicitly ask to create the draft in this turn, ask for confirmation.
Create the draft with GitHub-generated notes:
gh release create "$release_tag" -R "$repo" --draft --generate-notes --notes-start-tag "$previous_tag" --title "$release_title"
Add --verify-tag when the release must use an existing remote tag.
Organize and save the draft body:
tmp_dir="$(mktemp -d)"
gh release view "$release_tag" -R "$repo" --json body --jq '.body' > "$tmp_dir/generated.md"
node .agents/skills/create-draft-release-notes/scripts/create-draft-release-notes.mjs "$tmp_dir/generated.md" > "$tmp_dir/organized.md"
gh release edit "$release_tag" -R "$repo" --draft --title "$release_title" --notes-file "$tmp_dir/organized.md"
Return the draft URL:
gh release view "$release_tag" -R "$repo" --json url --jq '.url'
Use this when the user provides generated release note Markdown and only wants it organized:
node .agents/skills/create-draft-release-notes/scripts/create-draft-release-notes.mjs release-notes.md
Omit the file path to read from stdin. Review that every original item still appears once and non-item sections remain.
Emit non-empty sections in this order:
### Breaking Changes 🍭### New Features 🎉### Performance 🚀### Bug Fixes 🐞### Refactor 🔨### Document 📖### Other ChangesClassify by the item prefix:
type!: or type(scope)!:, plus breaking: / break:.feat: / feat(scope):, plus feature:.perf:.fix:.refactor:.docs: / docs(scope):, plus doc:.Keep each category in generated top-to-bottom order.
**Full Changelog**, or other non-item sections.scripts/create-draft-release-notes.mjs: deterministic formatter for generated release note Markdown.