name: release-fetch-changelog
description: >
Calls the GitHub REST API via gh to generate release notes for the given tag
(the same content the "Generate release notes" button produces on the web UI)
and inserts the result into the matching CHANGELOG-X.Y.md file. Used standalone
or by the /release orchestrator when UpdateChangelogReleaseWorker is up next.
user_invocable: true
version: 2.0.0
release-fetch-changelog
Automates the GitHub "Generate release notes" copy-paste step using the gh CLI. The GitHub-native generator stays the source of truth — this skill does not draft the changelog body itself.
Invocation
/release-fetch-changelog <version> --target-branch <branch> [--changelog-file <path>] [--dry-run]
<version>— e.g.v19.1.0,v19.0.1-rc1.--target-branch <branch>— e.g.19.0. Required.--changelog-file <path>— override the conventional locationCHANGELOG-<major>.<minor>.mdat repo root.--dry-run— fetch + show the generated markdown but do not write.
Pre-flight
Abort with a clear error if any fail:
gh auth statussucceeds.- The CHANGELOG file exists at the resolved path. If not, ask the operator to confirm the path with
--changelog-file. gh release view <version>returns "release not found" — bail out if the tag/release already exists.
Steps
- Find previous tag. Pick the highest version-sorted tag strictly less than
<version>, not the most-recently-published one. Shopsys maintains older majors as LTS lines (e.g.v14.5.1was published afterv18.0.0), so sorting bypublishedAtwould compare a new major against an older LTS patch and produce a misleading diff. Inject<version>into the tag list, sort, and print the entry immediately above it (substitute<version>with the actual version literal — leaving the placeholder string in place will produce no output because it fails the semver regex):{ echo "<version>"; gh api repos/shopsys/shopsys/tags --paginate --jq '.[].name'; } \ | grep -E '^v[0-9]+\.[0-9]+\.[0-9]+$' \ | sort -V -u \ | awk -v t="<version>" '$0 == t { print prev; exit } { prev = $0 }'sort -Valready orders semver tags ascending, so awk just walks the list and prints the line right before<version>. Theecho "<version>"injection makes this work whether or not<version>is already in the tag list. Forv19.0.0, this returnsv18.0.0, notv14.5.1. This is what GitHub's generator uses as the comparison baseline. - Generate notes via
gh. Call:
This returns the same markdown the "Generate release notes" button produces ongh api repos/shopsys/shopsys/releases/generate-notes \ -f tag_name=<version> \ -f previous_tag_name=<previous-tag> \ -f target_commitish=<branch> \ --jq .bodyhttps://github.com/shopsys/shopsys/releases/new. - Read the CHANGELOG. Find the first existing
## vheading. New content goes immediately above it. If no such heading exists, insert just after the line<!-- Add generated changelog below this line -->(the convention used byCHANGELOG-X.Y.mdheaders); if neither exists, prepend to the top. - Build the inserted block. A
## [<version>](https://github.com/shopsys/shopsys/compare/<previous-tag>...<version>) (YYYY-MM-DD)heading (today's date) + a blank line + the generated markdown + a trailing blank line. The bracketed/linked heading form is required —CheckChangelogForTodaysDateReleaseWorkermatches the regex#\#\# \[<version>\]\(.*\) \(\d+-\d+-\d+\)#and fails out of the release stage if the heading is a plain## <version> (date). - Write. Update the CHANGELOG file. Do not run
git addor commit —UpdateChangelogReleaseWorkerrunsphing markdown-fixand commits afterwards. - Report. Print:
- Inserted heading.
- First 3 lines of the generated body.
- Byte count.
- Path written.
On --dry-run, do steps 1-2 and print what would be inserted, then exit without writing.
Failure handling
gh apierror / non-2xx response → leave the CHANGELOG untouched, print the response body, instruct the operator to openhttps://github.com/shopsys/shopsys/releases/new?tag=<version>&target=<branch>manually and paste themselves. The/releaseorchestrator should detect this exit code and fall back to a regular surface-the-prompt flow.ghnot authenticated → tell the operator to rungh auth login. Do not attempt to authenticate for them.- Generated markdown is empty (no PRs between tags) → write
_No changes since previous tag._as the body and print a warning so the operator can double-check.
Constraints
- Never commit, push, or tag.
- Never edit any file other than the resolved CHANGELOG path.
- Never post to GitHub on the operator's behalf —
releases/generate-notesis a read-only generator endpoint; do not switch toPOST /releasesor any other write endpoint.