chore(go): route release notes through vociferate
This commit is contained in:
@@ -14,6 +14,7 @@ func main() {
|
||||
date := flag.String("date", "", "release date in YYYY-MM-DD format")
|
||||
recommend := flag.Bool("recommend", false, "print the recommended next release tag based on the changelog")
|
||||
printUnreleased := flag.Bool("print-unreleased", false, "print the current Unreleased changelog body")
|
||||
printReleaseNotes := flag.Bool("print-release-notes", false, "print the release notes section for --version")
|
||||
root := flag.String("root", ".", "repository root to update")
|
||||
versionFile := flag.String("version-file", "", "path to the version file, relative to --root")
|
||||
versionPattern := flag.String("version-pattern", "", "regexp with one capture group for the version value")
|
||||
@@ -52,8 +53,18 @@ func main() {
|
||||
return
|
||||
}
|
||||
|
||||
if *printReleaseNotes {
|
||||
body, err := vociferate.ReleaseNotes(absRoot, *version, opts)
|
||||
if err != nil {
|
||||
fmt.Fprintf(os.Stderr, "print release notes: %v\n", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
fmt.Print(body)
|
||||
return
|
||||
}
|
||||
|
||||
if *version == "" || *date == "" {
|
||||
fmt.Fprintln(os.Stderr, "usage: vociferate --version <version> --date <YYYY-MM-DD> [--root <dir>] [--version-file <path>] [--version-pattern <regexp>] [--changelog <path>] | --recommend [--root <dir>] [--version-file <path>] [--version-pattern <regexp>] [--changelog <path>] | --print-unreleased [--root <dir>] [--changelog <path>]")
|
||||
fmt.Fprintln(os.Stderr, "usage: vociferate --version <version> --date <YYYY-MM-DD> [--root <dir>] [--version-file <path>] [--version-pattern <regexp>] [--changelog <path>] | --recommend [--root <dir>] [--version-file <path>] [--version-pattern <regexp>] [--changelog <path>] | --print-unreleased [--root <dir>] [--changelog <path>] | --print-release-notes --version <version> [--root <dir>] [--changelog <path>]")
|
||||
os.Exit(2)
|
||||
}
|
||||
|
||||
|
||||
@@ -215,6 +215,11 @@ func UnreleasedBody(rootDir string, options Options) (string, error) {
|
||||
return defaultService().UnreleasedBody(rootDir, options)
|
||||
}
|
||||
|
||||
// ReleaseNotes returns the release section for a specific semantic version.
|
||||
func ReleaseNotes(rootDir, version string, options Options) (string, error) {
|
||||
return defaultService().ReleaseNotes(rootDir, version, options)
|
||||
}
|
||||
|
||||
// RecommendedTag returns the next semantic release tag based on current changelog state.
|
||||
func (s *Service) RecommendedTag(rootDir string, options Options) (string, error) {
|
||||
resolved, err := resolveOptions(options)
|
||||
@@ -281,6 +286,16 @@ func (s *Service) UnreleasedBody(rootDir string, options Options) (string, error
|
||||
return s.readUnreleasedBody(rootDir, resolved.Changelog)
|
||||
}
|
||||
|
||||
// ReleaseNotes returns the release section for a specific semantic version.
|
||||
func (s *Service) ReleaseNotes(rootDir, version string, options Options) (string, error) {
|
||||
resolved, err := resolveOptions(options)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
return s.readReleaseNotes(rootDir, strings.TrimPrefix(strings.TrimSpace(version), "v"), resolved.Changelog)
|
||||
}
|
||||
|
||||
func sectionHasEntries(unreleasedBody, sectionName string) bool {
|
||||
heading := "### " + sectionName
|
||||
sectionStart := strings.Index(unreleasedBody, heading)
|
||||
@@ -471,6 +486,10 @@ func readLatestChangelogVersion(rootDir, changelogPath string) (string, bool, er
|
||||
return defaultService().readLatestChangelogVersion(rootDir, changelogPath)
|
||||
}
|
||||
|
||||
func readReleaseNotes(rootDir, version, changelogPath string) (string, error) {
|
||||
return defaultService().readReleaseNotes(rootDir, version, changelogPath)
|
||||
}
|
||||
|
||||
func (s *Service) readLatestChangelogVersion(rootDir, changelogPath string) (string, bool, error) {
|
||||
path := filepath.Join(rootDir, changelogPath)
|
||||
contents, err := s.fileSystem.ReadFile(path)
|
||||
@@ -485,6 +504,38 @@ func (s *Service) readLatestChangelogVersion(rootDir, changelogPath string) (str
|
||||
return match[1], true, nil
|
||||
}
|
||||
|
||||
func (s *Service) readReleaseNotes(rootDir, version, changelogPath string) (string, error) {
|
||||
if version == "" {
|
||||
return "", fmt.Errorf("release version must not be empty")
|
||||
}
|
||||
|
||||
path := filepath.Join(rootDir, changelogPath)
|
||||
contents, err := s.fileSystem.ReadFile(path)
|
||||
if err != nil {
|
||||
return "", fmt.Errorf("read changelog: %w", err)
|
||||
}
|
||||
|
||||
text := string(contents)
|
||||
headingExpr := regexp.MustCompile(`(?m)^## \[` + regexp.QuoteMeta(version) + `\](?:\([^\n)]*\))? - `)
|
||||
headingLoc := headingExpr.FindStringIndex(text)
|
||||
if headingLoc == nil {
|
||||
return "", fmt.Errorf("release notes section for %s not found in changelog", version)
|
||||
}
|
||||
|
||||
nextSectionRelative := strings.Index(text[headingLoc[0]+1:], "\n## [")
|
||||
sectionEnd := len(text)
|
||||
if nextSectionRelative != -1 {
|
||||
sectionEnd = headingLoc[0] + 1 + nextSectionRelative
|
||||
}
|
||||
|
||||
section := text[headingLoc[0]:sectionEnd]
|
||||
if !strings.HasSuffix(section, "\n") {
|
||||
section += "\n"
|
||||
}
|
||||
|
||||
return section, nil
|
||||
}
|
||||
|
||||
func deriveRepositoryURL(rootDir string) (string, bool) {
|
||||
return defaultService().deriveRepositoryURL(rootDir)
|
||||
}
|
||||
|
||||
@@ -36,6 +36,19 @@ outputs:
|
||||
runs:
|
||||
using: composite
|
||||
steps:
|
||||
- name: Resolve vociferate binary metadata
|
||||
id: resolve-binary
|
||||
shell: bash
|
||||
env:
|
||||
ACTION_REF: ${{ github.action_ref }}
|
||||
ACTION_REPOSITORY: ${{ github.action_repository }}
|
||||
CACHE_TOKEN: ${{ env.VOCIFERATE_CACHE_TOKEN }}
|
||||
SERVER_URL: ${{ github.server_url }}
|
||||
RUNNER_ARCH: ${{ runner.arch }}
|
||||
RUNNER_TEMP: ${{ runner.temp }}
|
||||
run: |
|
||||
bash "$GITHUB_ACTION_PATH/../script/resolve-vociferate-runtime.sh"
|
||||
|
||||
- name: Resolve release version
|
||||
id: resolve-version
|
||||
shell: bash
|
||||
@@ -63,9 +76,53 @@ runs:
|
||||
echo "tag=$tag" >> "$GITHUB_OUTPUT"
|
||||
echo "version=$normalized" >> "$GITHUB_OUTPUT"
|
||||
|
||||
- name: Extract release notes from changelog
|
||||
id: extract-notes
|
||||
- name: Setup Go
|
||||
if: steps.resolve-binary.outputs.use_binary != 'true'
|
||||
uses: actions/setup-go@v5
|
||||
with:
|
||||
go-version: '1.26.1'
|
||||
cache: true
|
||||
cache-dependency-path: ${{ steps.resolve-binary.outputs.source_root }}/go.sum
|
||||
|
||||
- name: Restore cached vociferate binary
|
||||
id: cache-vociferate
|
||||
if: steps.resolve-binary.outputs.use_binary == 'true'
|
||||
uses: actions/cache@v4
|
||||
with:
|
||||
path: ${{ steps.resolve-binary.outputs.cache_dir }}
|
||||
key: vociferate-${{ steps.resolve-binary.outputs.cache_token }}-linux-${{ runner.arch }}
|
||||
|
||||
- name: Download vociferate binary
|
||||
if: steps.resolve-binary.outputs.use_binary == 'true' && steps.cache-vociferate.outputs.cache-hit != 'true'
|
||||
shell: bash
|
||||
env:
|
||||
TOKEN: ${{ github.token }}
|
||||
ASSET_URL: ${{ steps.resolve-binary.outputs.asset_url }}
|
||||
BINARY_PATH: ${{ steps.resolve-binary.outputs.binary_path }}
|
||||
run: |
|
||||
bash "${{ steps.resolve-binary.outputs.source_root }}/script/download-vociferate-binary.sh"
|
||||
|
||||
- name: Extract release notes from binary
|
||||
id: extract-notes-binary
|
||||
if: steps.resolve-binary.outputs.use_binary == 'true'
|
||||
shell: bash
|
||||
env:
|
||||
VOCIFERATE_BIN: ${{ steps.resolve-binary.outputs.binary_path }}
|
||||
CHANGELOG: ${{ inputs.changelog != '' && inputs.changelog || 'CHANGELOG.md' }}
|
||||
RELEASE_VERSION: ${{ steps.resolve-version.outputs.version }}
|
||||
RUNNER_TEMP: ${{ runner.temp }}
|
||||
run: |
|
||||
set -euo pipefail
|
||||
|
||||
notes_file="${RUNNER_TEMP}/release-notes.md"
|
||||
"$VOCIFERATE_BIN" --print-release-notes --version "$RELEASE_VERSION" --root "$GITHUB_WORKSPACE" --changelog "$CHANGELOG" > "$notes_file"
|
||||
printf 'notes_file=%s\n' "$notes_file" >> "$GITHUB_OUTPUT"
|
||||
|
||||
- name: Extract release notes from source
|
||||
id: extract-notes-source
|
||||
if: steps.resolve-binary.outputs.use_binary != 'true'
|
||||
shell: bash
|
||||
working-directory: ${{ steps.resolve-binary.outputs.source_root }}
|
||||
env:
|
||||
CHANGELOG: ${{ inputs.changelog != '' && inputs.changelog || 'CHANGELOG.md' }}
|
||||
RELEASE_VERSION: ${{ steps.resolve-version.outputs.version }}
|
||||
@@ -73,22 +130,25 @@ runs:
|
||||
run: |
|
||||
set -euo pipefail
|
||||
|
||||
release_notes="$(awk -v version="$RELEASE_VERSION" '
|
||||
$0 ~ "^## \\[" version "\\]" {capture=1}
|
||||
capture {
|
||||
if ($0 ~ "^## \\[" && $0 !~ "^## \\[" version "\\]") exit
|
||||
print
|
||||
}
|
||||
' "$CHANGELOG")"
|
||||
notes_file="${RUNNER_TEMP}/release-notes.md"
|
||||
go run ./cmd/vociferate --print-release-notes --version "$RELEASE_VERSION" --root "$GITHUB_WORKSPACE" --changelog "$CHANGELOG" > "$notes_file"
|
||||
printf 'notes_file=%s\n' "$notes_file" >> "$GITHUB_OUTPUT"
|
||||
|
||||
if [[ -z "${release_notes//[[:space:]]/}" ]]; then
|
||||
echo "Release notes section for ${RELEASE_VERSION} was not found in ${CHANGELOG}" >&2
|
||||
exit 1
|
||||
- name: Finalize release notes file
|
||||
id: extract-notes
|
||||
shell: bash
|
||||
env:
|
||||
NOTES_FILE_BINARY: ${{ steps.extract-notes-binary.outputs.notes_file }}
|
||||
NOTES_FILE_SOURCE: ${{ steps.extract-notes-source.outputs.notes_file }}
|
||||
run: |
|
||||
set -euo pipefail
|
||||
|
||||
notes_file="$NOTES_FILE_BINARY"
|
||||
if [[ -z "$notes_file" ]]; then
|
||||
notes_file="$NOTES_FILE_SOURCE"
|
||||
fi
|
||||
|
||||
notes_file="${RUNNER_TEMP}/release-notes.md"
|
||||
printf '%s\n' "$release_notes" > "$notes_file"
|
||||
echo "notes_file=$notes_file" >> "$GITHUB_OUTPUT"
|
||||
printf 'notes_file=%s\n' "$notes_file" >> "$GITHUB_OUTPUT"
|
||||
|
||||
- name: Create or update release
|
||||
id: create-release
|
||||
|
||||
9
script/download-vociferate-binary.sh
Normal file
9
script/download-vociferate-binary.sh
Normal file
@@ -0,0 +1,9 @@
|
||||
#!/usr/bin/env bash
|
||||
|
||||
set -euo pipefail
|
||||
|
||||
curl --fail --location \
|
||||
-H "Authorization: token ${TOKEN}" \
|
||||
-o "$BINARY_PATH" \
|
||||
"$ASSET_URL"
|
||||
chmod +x "$BINARY_PATH"
|
||||
51
script/resolve-vociferate-runtime.sh
Normal file
51
script/resolve-vociferate-runtime.sh
Normal file
@@ -0,0 +1,51 @@
|
||||
#!/usr/bin/env bash
|
||||
|
||||
set -euo pipefail
|
||||
|
||||
case "${RUNNER_ARCH}" in
|
||||
X64)
|
||||
arch="amd64"
|
||||
;;
|
||||
ARM64)
|
||||
arch="arm64"
|
||||
;;
|
||||
*)
|
||||
echo "Unsupported runner architecture: ${RUNNER_ARCH}" >&2
|
||||
exit 1
|
||||
;;
|
||||
esac
|
||||
|
||||
source_root="${GITHUB_ACTION_PATH}"
|
||||
if [[ ! -f "${source_root}/go.mod" ]]; then
|
||||
source_root="$(realpath "${GITHUB_ACTION_PATH}/..")"
|
||||
fi
|
||||
|
||||
printf 'source_root=%s\n' "$source_root" >> "$GITHUB_OUTPUT"
|
||||
|
||||
if [[ "${ACTION_REF:-}" == v* ]]; then
|
||||
release_tag="${ACTION_REF}"
|
||||
normalized_version="${release_tag#v}"
|
||||
asset_name="vociferate_${normalized_version}_linux_${arch}"
|
||||
cache_dir="${RUNNER_TEMP}/vociferate/${release_tag}/linux-${arch}"
|
||||
binary_path="${cache_dir}/vociferate"
|
||||
asset_url="${SERVER_URL}/aether/vociferate/releases/download/${release_tag}/${asset_name}"
|
||||
|
||||
provided_cache_token="$(printf '%s' "${CACHE_TOKEN:-}" | sed 's/^[[:space:]]\+//; s/[[:space:]]\+$//')"
|
||||
if [[ -n "$provided_cache_token" ]]; then
|
||||
cache_token="$provided_cache_token"
|
||||
else
|
||||
cache_token="${ACTION_REPOSITORY:-aether/vociferate}-${release_tag}"
|
||||
fi
|
||||
|
||||
mkdir -p "$cache_dir"
|
||||
|
||||
printf 'use_binary=true\n' >> "$GITHUB_OUTPUT"
|
||||
printf 'release_tag=%s\n' "$release_tag" >> "$GITHUB_OUTPUT"
|
||||
printf 'cache_token=%s\n' "$cache_token" >> "$GITHUB_OUTPUT"
|
||||
printf 'asset_name=%s\n' "$asset_name" >> "$GITHUB_OUTPUT"
|
||||
printf 'asset_url=%s\n' "$asset_url" >> "$GITHUB_OUTPUT"
|
||||
printf 'cache_dir=%s\n' "$cache_dir" >> "$GITHUB_OUTPUT"
|
||||
printf 'binary_path=%s\n' "$binary_path" >> "$GITHUB_OUTPUT"
|
||||
else
|
||||
printf 'use_binary=false\n' >> "$GITHUB_OUTPUT"
|
||||
fi
|
||||
Reference in New Issue
Block a user